Replicating and improving the lightbox module seen on Medium with fluid transitions. View demo here. For users who are looking for a quick setup and/or troubleshooting process, refer to basic usage, but do not forget to read the usage precautions and frequently asked questions.
Although not thoroughly tested, Fluidbox should be working in IE ≥10 and all versions of Chrome, Firefox, Safari, iOS Safari and Android Chrome, with the exception of Opera Mini. However, I suggest disabling Fluidbox on mobile devices or at small screen resolutions.
Special thanks to the following stellar folks who has helped majorly in making Fluidbox better:
- @hybernaut for refactoring the code and reorganizing functions
- @maxee for implementation of a new feature that enables differential image ratios between thumbnails and linked image
In addition, a shoutout to:
- Paul Irish for graciously allowing me to include a simple debouncing script in the plugin to handle viewport resize events
- David Walsh and Jonathan Suh for their insight on listening to
transitionend
events
Fluidbox was initially a simple personal challenge I set myself, with two simple aims — to replicate the distraction-free, fluid lightbox seen on Medium, and to improve on it such that it will allow linking to a larger-resolution image. The plugin deals with higher resolution, linked images elegantly, such that it only preloads them when users click on the thumbnails, therefore conserving bandwidth usage for your visitors and your server(s).
The plugin is relatively lightweight: 5.38kb (1.81kb after gzipped) for the minified JS file, and 1.02kb (369b after gzipped) for the minimal stylesheet.
You can read my article on Medium about how I got inspiration for this little project of mine, and the basic mechanisms behind the plugin. Some serious math is involved (nah, not really).
Moreover, you can visit the demo of this plugin on the project page hosted with GitHub. The plugin version 1.22 and onwards (uncompressed, minified and its associated CSS file) is hosted with CDNJS.
Fluidbox is part of the vast collection of libraries proudly hosted by CDNJS. You can reference all versions of Fluidbox published hitherto from there.
Fluidbox has been implemented on other sites in the wild, too — check it out:
- Gemma Busquets by @imgemmabusquets
- Terry Mun by myself
Some external resources that might help you:
To add your site that has implemented Fluidbox, or an article/tutorial you have written on Fluidbox use and/or application, feel free to write to me at @teddyrised.
Version | Comments |
---|---|
1.2.0 | Official release |
1.2.1 | Minor bug fixes |
1.2.2 |
|
1.2.3 |
|
1.2.4α | Warning: Buggy, alpha-build
|
1.2.5 |
|
1.2.6 |
|
1.2.7 |
|
1.3.0 |
|
1.3.1 |
|
1.3.2 |
|
1.3.3 |
|
1.3.4 |
|
1.3.5 |
|
1.4.0 |
|
1.4.1 |
|
1.4.2 |
|
To install Fluidbox, you will have to include the following resources in your page. The JS files should be loaded in the order stipulated below. For the CSS file, you can either incorporate it with your site's stylesheet, or load it externally through the <link>
element in <head>
.
Type | File Name | Description |
---|---|---|
JS | jQuery 1.x | External Dependency: The latest verson of jQuery 1.x library is needed for Fluidbox functionality. Minimum version requirement: v1.7 |
JS | jquery.fluidbox.js |
Confers the main functionality of Fluidbox. Alternatively, you can load the minified version, jquery.fluidbox.min.js |
CSS | css/fluidbox.css |
Offers styles that are crucial for the correct display of Fluidbox. The appearance will break if this is not included. |
It is rather straightforward to use Fluidbox — simply chain the .fluidbox()
method to a selector of your choice. The plugin will automatically check if the selector is:
- An anchor element with an href to the correct image url
- Contains one and only one child
- The only children is an
<img>
element - Is visible upon DOM ready (v1.3.4 onwards)
In the event that the element that satisfies the selector criteria but failed any one of the above criteria, the element will be ignored and the plugin moves on to the next available element. Therefore, it is important that your Fluidbox element(s) follow the following format. The title
and alt
attributes of the <img>
element is not used by the plugin, but the alt
attribute has to be present for it to be semantically valid.
<a href="path/to/image">
<img src="path/to/image" alt="" />
</a>
In your JS file, you can simply chain the .fluidbox()
method to your selector on DOM ready:
$(function () {
$('a').fluidbox();
})
The selector can be anything of your choice. Let's say you want to target the <a>
elements specifically in a certain section on your page:
<section id="gallery">
<h1>Title</h1>
<p>Introductory text with an <a href="#">unrelated link</a></p>
<a href="..." rel="lightbox">
<img src="..." alt="" />
</a>
</section>
Then, you can use, for example:
$(function () {
$('#gallery a[rel="lightbox"]').fluidbox();
})
As of v1.4.2, Fluidbox allows users to trigger manual recomputing of the dimensions of the overlay and that of the active Fluidbox instance. This functionality is extremely useful when you make layout changes that repaints the canvas and alters the dimensions of linked Fluidbox images. In order to do so, you will have to trigger the event manually, after the changes in the DOM, or element dimensions has been applied.
An example would be changing the size of the Fluidbox thumbnail:
$('button').click(function() {
$(selector)
.css('width', '200px')
.trigger('recompute');
});
As of v1.4.1, Fluidbox will trigger several distinct events depending on the state of the current (and only) instance of Fluidbox. You should use .on()
to listen to the event being triggered, and can add your own custom callbacks if necessary, for example:
$(selector)
.fluidbox()
.on('openstart', doSomething)
.on('closeend', doSomethingElse);
The list of custom events supported by Fluidbox is as follow:
Event | Version | Description |
---|---|---|
openstart |
1.4.1 | Fired when a click event is registered from a Fluidbox instance that triggers its opening. This is called after the linked image has been successfully loaded. |
openend |
1.4.1 | Fired when the transitionend event is fired (with appropriate vendors supported). This happens when Fluidbox has been scaled to its final size (determined by viewportScale , see configuration). The timing between openstart and openend are dictated by the transition-duration settings in fluidbox.css , or any overrides that you have implemented that targets the class .fluidbox-ghost . |
resizeend |
1.4.1 | Fired when the positioning and scale of an opened Fluidbox instance is recalculated and has been transitioned to completion. |
closestart |
1.4.1 | Fired when a click event is registered from a Fluidbox instance that triggers its closing. |
closeend |
1.4.1 | Fired when the transitionend event is fired (with appropriate vendors supported). This happens when Fluidbox has been scaled back to its original thumbnail size on the page. The timing between closestart and closeend are dictated by the transition-duration settings in fluidbox.css , or any overrides that you have implemented that targets the class .fluidbox-ghost . |
recomputeend |
1.4.2 | Fired when the Fluidbox ghost element, or the active Fluidbox on display, is recomputed due to layout changes not dependent on the $(window).resize() event. This is triggered manually by the custom trigger recompute (see usage instructions). |
Previously hidden elements
As of v1.3.4, Fluidbox will only work with elements that are visible, i.e. not display: none
, on the page upon DOM ready. This is because dimensions of hidden images (or images in parents who are hidden) are inaccesible to Fluidbox, resulting in an error. You will have to rebind Fluidbox to the newly revealted elements. Given the example below:
// Apply Fluidbox to elements of interest
$('.gallery a').fluidbox();
// User-triggered event to display gallery
$('#show-gallery').click(function () {
$(this).next().show();
});
<button type="button" id="show-gallery">Show Gallery</button>
<div class="gallery" style="display: none">
<a href="...">
<img src="..." alt="" />
</a>
</div>
You will realize that, even after revealing the element, the Fluidbox method is not working for it. That is because non-visible elements, despite satisfying the selector, will not be bound. So, use the following code instead:
// Apply Fluidbox to elements of interest
$('.gallery a').fluidbox();
// User-triggered event to display gallery
$('#show-gallery').click(function () {
$(this)
.next()
.show() // Show gallery
.find('a') // Fluidbox to all elements in gallery
.fluidbox();
});
In order to enable Fluidbox functionality to dynamically-added content, you will have to apply .fluidbox()
to the element of interest after appending it to the DOM. For example, let's say clicking a <button>
triggers the addition of a new image:
$(function() {
$('button').click(function(e) {
e.preventDefault();
// Construct new image
var $newContent = $('<div><p>This is a new image that is dynamically-added to the page.</p><a href="http://placehold.it/500x500" title="" data-fluidbox><img src="http://placehold.it/200x200" alt="Alternate Text" title="Image Title" /></a></div>');
// Insert new content object into DOM, and then apply .fluidbox to it
$(this).after($newContent).next().find('a[data-fluidbox]').fluidbox();
});
));
You are of course welcome to use other ways to manipulate and/or transverse the DOM, but you will have to adjust the search/filter function (using .find()
or other similar jQuery methods) to retrieve the newly inserted content and search for the element of interest where you want Fluidbox to work with.
Fluidbox can be configured according to your needs. The following options are available:
Variable/Option | Type | Default value | Description |
---|---|---|---|
viewportFill |
Numerical | 0.95 |
Dictates how much the longest axis of the image should fill the viewport. The default value will make the image fill 95% of the viewport dimension along its longest axis |
overlayColor |
String | rgba(255,255,255,.85) |
Warning: Deprecated from v1.3.5. The overlay color is now dictated in the stylesheet, allowing custom overlays for each Fluidbox instance. Legacy description: Sets the |
debounceResize |
Boolean | true |
Dictates if the $(window).resize() event should be debounced for performance reason. This feature leverages the small snippet kindly provided by Paul Irish. |
closeTrigger |
Array | see below | Dictates what event triggers closing of an opened Fluidbox. The default setup binds the click handler to the overlay. |
stackIndex |
Integer | 999 |
Determines how high up the z-index will all Fluildbox elements be. Leave this option as default, unless you have other relatively or absolutely positioned elements on the page that is messing with Fluidbox appearance. |
User-defined settings have to be passed as the aforementioned variables/options to the .fluidbox()
method, i.e.:
$('a').fluidbox({
viewportFill: 0.8,
debounceResize: false
});
The default setup will have the effect of binding the click event to the overlay, so that when user click on the overlay, the Fluidbox instance that is opened will be closed:
// Default option
closeTrigger: [
{
selector: '#fluidbox-overlay',
event: 'click'
},
{
selector: 'document',
event: 'keyup',
keyCode: 27
}
]
It is also possible to bind other events to trigger the same effect. For example, if you would want to close the Fluidbox when the viewport is resized, you can do the following:
$(function () {
$('a').fluidbox({
closeTrigger: [
{ selector: '#fluidbox-overlay', event: 'click' },
{ selector: 'window', event: 'resize' }
]
});
});
This will have the effect of doing so (where closeFb
is the internal function in the plugin needed to close any opened Fluidbox):
$(document).on('click', '#fluidbox-overlay', closeFb);
$(window).on('resize', closeFb);
You can even bind event to multiple selectors, and vice versa. The syntax of dictating so is similar to constructing event handler binding using the .on()
method, so if you are familiar with its use, dictating your own closeTrigger should not be too difficult:
$(function () {
$('a').fluidbox({
closeTrigger: [
{ selector: '#fluidbox-overlay', event: 'click' },
{ selector: 'window', event: 'resize scroll' },
{ selector: '#ele1, #ele2', event: 'hover' }
]
});
});
This will have the effect of doing so:
$(document).on('click', '#fluidbox-overlay', closeFb);
$(window).on('resize scroll', closeFb);
$(document).on('hover', '#ele1, #ele2', closeFb)
Fluidbox require the following dependencies in order to function properly — you will have to include them in your page, if you want Fluidbox to work:
- The latest release of jQuery 1.x (minimum requirement: jQuery ≥1.7), available from Google's jQuery API at
http://ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js
Note that the imagesloaded jQuery plugin is no longer required as of v1.2.8 and above.
Fluidbox allows you to throttle the $(window).resize()
event, and this is only possible with Paul Irish's debounced resize function. The small script has been included in the plugin by default, but I would like to extend my gratitude to Paul for making it available, and for allowing me to include it in this plugin.
Fluidbox may not work properly in the event that you have set your parent container, or content wrapping elements, such as <div id="#content">
and the likes, to hide their overflowing content, i.e. overflow: hidden
. This is because the enlarged image is positioned relatively to its hyperlink, and not absolutely or fixed within the viewport.
When you want an absolutely/fixed-positioned element on the page to not be obscured by the dynamically-generated wrapping element, you should use a z-index of between 1000 to 1010. 1000 is set as the default stackIndex
of Fluidbox, while 10 is set as the default stackIndexDelta
, which is toggled on/off depending on the state of the Fluidbox. These settings can be individually tuned, see Configuration below.
Binding Fluidbox to previously hidden images
As Fluidbox requires access to the final calculation dimensions of the image in question in order to (1) position the ghost element correctly and (2) calculate the correct scale factor and transform values, it will only bind to images that are visible upon DOM ready. If you are relying on dynamic events (e.g. user-triggered, AJAX-loaded and etc.) to trigger a later appearance of an image, rebind Fluidbox to freshly revealed elements. This also applies to dynamically-loaded content, see demo for a working example.
-
Fluidbox is not working in my installation. Where should I start?
Start by checking your browser's console log. What error messages do you see? Also, make sure that you are using the latest version of jQuery 1.x (minimum requirement: v1.8 or above) and that the dependencies have been loaded successfully. Also, did you remember reading the usage precautions? You might have encountered a scenario where Fluidbox is not designed to handle. -
Do you plan to implement [insert feature]?
Fluidbox is conceived as a means to simplify lightboxes. Therefore, I plan to keep Fluidbox as simple as it is, without additional features, such as captioning (there are other limitations to this, too) and gallery features. However, you can always fork and modify Fluidbox to your personal liking. Manual captioning is possible, refer to the [advanced demo]. -
The image url isn't being interpretted correctly.
Fluidbox fetches the larger image based on the URL specified in thehref
attribute if the wrapping anchor (<a>
) tag. -
I have a application-specific problem that I need help troubleshooting. Can you help me?
Of course! I am more than happy to help, but it really depends if you have a clear problem statement and a demo that I can play around with — I strongly encourage you to host your reduced test case(s) with either JSFiddle, CodePen or the likes. Then, create a new issue. I promise I will get back to you when I have time. -
Do you provide private support by email / phone call / Skype call / (insert any other forms of communication)?
Since Fluidbox is provided as-is and free-of-charge, I am sorry to inform you that it is so far not possible for me to dedicate so much effort. However, you can follow what is described in step #4.
This plugin is licensed under the MIT License.