Muuri creates responsive, sortable, filterable and draggable grid layouts. Yep, that's a lot of features in one library, but we have tried to make it as tiny as possible. Comparing to what's out there Muuri is a combination of Packery, Masonry, Isotope and jQuery UI sortable. Wanna see it in action? Check out the demo on the website.
Muuri's layout system allows positioning the grid items within the container in pretty much any way imaginable. The default "First Fit" bin packing layout algorithm generates similar layouts as Packery and Masonry. The implementation is heavily based on the "maxrects" approach as described by Jukka Jylänki in his research A Thousand Ways to Pack the Bin. However, you can also provide your own layout algorithm to position the items in any way you want.
Muuri uses Velocity for animating the grid items (positioining/showing/hiding) and Hammer.js for handling the dragging. Hammer.js is an optional dependency that is only required if dragging is enabled, but Velocity is a hard dependency.
And if you're wondering about the name of the library "muuri" is Finnish meaning a wall.
A word of warning. Muuri is currently under active development and might be still a bit rough on the edges so production use is not recommended just yet. But when did that ever stop you ;)
- Getting started
- Options
- Methods
- muuri.on( event, listener )
- muuri.off( event, listener )
- muuri.refresh( [targets] )
- muuri.get( [targets], [state] )
- muuri.add( elements, [index] )
- muuri.remove( targets, [removeElement] )
- muuri.synchronize()
- muuri.layout( [instant], [callback] )
- muuri.show( targets, [instant], [callback] )
- muuri.hide( targets, [instant], [callback] )
- muuri.indexOf( target )
- muuri.move( targetFrom, targetTo )
- muuri.swap( targetA, targetB )
- muuri.destroy()
- Events
- License
Muuri depends on the following libraries:
First, include Muuri and it's dependencies in your site.
<script src="velocity.js"></script>
<script src="hammer.js"></script>
<!-- Needs to be within in body element or have access to body element -->
<script src="muuri.js"></script>
An important note for including Muuri to your site is that it needs to have access to the body
element when it's loaded. Muuri does some feature checking on init and might not work correctly if it does not have access to the body
element.
Then, define your grid markup.
- Every grid must have a container element and item element(s) within the container element.
- A grid item must always consist of at least two elements. The outer element is used for positioning the item and the inner element (first direct child) is used for animating the item's visibility (show/hide methods). You can insert any markup you wish inside the inner item element.
<div class="grid">
<div class="item">
<div class="item-content">
This can be anything.
</div>
</div>
<div class="item">
<div class="item-content">
<div class="my-custom-content">
Yippee!
</div>
</div>
</div>
</div>
Next, let's apply some styles.
- The grid element must be "positioned" meaning that it's CSS position property must be set to relative, absolute or fixed. Also note that Muuri automatically resizes the container element depending on the area the items cover.
- The item elements must have their CSS position set to absolute and their display property set to block, unless of course the elements have their display set to block inherently.
- The item elements must not have any CSS transitions or animations applied to them since it might conflict with Velocity's animations.
- You can control the gaps between the tiles by giving some margin to the item elements.
.grid {
position: relative;
}
.item {
display: block;
position: absolute;
width: 100px;
height: 100px;
margin: 5px;
z-index: 1;
}
.item.muuri-dragging,
.item.muuri-releasing {
z-index: 2;
}
.item.muuri-hidden {
z-index: 0;
}
.item-content {
position: relative;
width: 100%;
height: 100%;
}
Finally, initiate a Muuri instance.
- The bare minimum configuration is demonstrated below. You must always provide Muuri with the container element and the initial item elements.
- Be sure to check out the all the available options, methods and events.
var grid = new Muuri({
container: document.getElementsByClassName('grid')[0],
// Muuri does not convert a node list to array automatically
// so we have to do it manually.
items: [].slice.call(document.getElementsByClassName('item'))
});
container
— element- Default value:
null
. - The container element. Must be always defined.
- Default value:
items
— array of elements- Default value:
[]
. - The initial item elements wrapped in an array. The elements must be children of the container element.
- Default value:
positionDuration
— number- Default value:
300
. - The duration for item's positioning animation in milliseconds. Set to
0
to disable.
- Default value:
positionEasing
— string / array- Default value:
"ease-out"
. - The easing for item's positioning animation. Read Velocity's easing documentation for more info on possible easing values.
- Default value:
show
— object- Default value:
{duration: 300, easing: "ease-out"}
. - The object should contain duration (integer, milliseconds) and easing properties. Set to null to disable hide animation altogether.
- Default value:
hide
— object- Default value:
{duration: 300, easing: "ease-out"}
. - The object should contain duration (integer, milliseconds) and easing properties. Set to null to disable hide animation altogether.
- Default value:
layout
— array / function / string- Default value:
"firstFit"
. - Define the layout method to be used for calculating the positions of the items. If you provide a string or an array Muuri will try to locate a registered layout method in
Muuri.Layout.methods
object. Currently there is only one built-in method:"firstFit"
. - The array syntax is the only way to use the built-in layout methods and provide arguments for them. The first value should be a string (name of the method) and the second value (optional) should be a configuration object, e.g.
["firstFit", {horizontal: true}]
. - You can always just provide a function which will receive a
Muuri.Layout
instance as it's context which you can manipulate as much as you want to get the items to the wanted positions. More info about rolling your own layout method is coming up later on, but in the meantime you can check the source code and see how the default method is implemented. - Available methods and related settings:
"firstFit"
horizontal
(type: boolean, default:false
)- When
true
the grid works in landscape mode (grid expands to the right). Use for horizontally scrolling sites. Whenfalse
the grid works in "portrait" mode and expands downwards.
- When
alignRight
(type: boolean, default:false
)- When
true
the items are aligned from right to left.
- When
alignBottom
(type: boolean, default:false
)- When
true
the items are aligned from the bottom up.
- When
fillGaps
(type: boolean, default:false
)- When
true
the algorithm goes through every item in order and places each item to the first available free slot, even if the slot happens to be visually before the previous element's slot. Practically this means that the items might not end up visually in order, but there will be less gaps in the grid. By default this options isfalse
which basically means that the following condition will be always true when calculating the layout:nextItem.top > prevItem.top || (nextItem.top === prevItem.top && nextItem.left > prevItem.left)
. This also means that the items will be visually in order.
- When
- Default value:
layoutOnResize
— null / number- Default value:
100
. - Should Muuri automatically trigger layout on window resize? Set to
null
to disable. When a number (0
or greater) is provided Muuri will automatically trigger layout when window is resized. The provided number equals to the amount of time (in milliseconds) that is waited before the layout is triggered after each resize event. The layout method is wrapped in a debouned function in order to avoid unnecessary layout calls.
- Default value:
layoutOnInit
— boolean- Default value:
true
. - Should Muuri trigger layout automatically on init?
- Default value:
dragEnabled
— boolean- Default value:
false
. - Should items be draggable?
- Default value:
dragContainer
— null / element- Default value:
null
. - Which item should the dragged item be appended to for the duration of the drag? If
null
is provided the item's muuri container element will be used.
- Default value:
dragPredicate
— null / function- Default value:
null
. - A function that determines when dragging should start. Set to null to use the default predicate.
- Default value:
dragSort
— boolean- Default value:
true
. - Should the items be sorted during drag?
- Default value:
dragSortInterval
— number- Default value:
50
. - When an item is dragged around the grid Muuri automatically checks if the item overlaps another item enough to move the item in it's place. The overlap check method is debounced and this option defines the debounce interval in milliseconds. In other words, this is option defines the amount of time the dragged item must be still before the overlap is checked.
- Default value:
dragSortTolerance
— number- Default value:
50
. - Allowed values:
1
-100
. - How many percent the intersection area between the dragged item and the compared item should be from the maximum potential intersection area between the two items in order to justify for sorting to occur.
- Default value:
dragSortAction
— string- Default value:
"move"
. - Allowed values:
"move"
,"swap"
. - Should the dragged item be moved to the new position or should it swap places with the item it overlaps?
- Default value:
dragReleaseDuration
— number- Default value:
300
. - The duration for item's drag release animation. Set to
0
to disable.
- Default value:
dragReleaseEasing
— string / array- Default value:
"ease-out"
. - The easing for item's drag release animation. Read Velocity's easing documentation for more info on possible easing values.
- Default value:
containerClass
— string- Default value:
"muuri"
. - Container element classname.
- Default value:
itemClass
— string- Default value:
"muuri-item"
. - Item element classname.
- Default value:
shownClass
— string- Default value:
"muuri-shown"
. - Visible item classname.
- Default value:
hiddenClass
— string- Default value:
"muuri-hidden"
. - Hidden item classname.
- Default value:
positioningClass
— string- Default value:
"muuri-positioning"
. - This classname will be added to the item element for the duration of positioing.
- Default value:
draggingClass
— string- Default value:
"muuri-dragging"
. - This classname will be added to the item element for the duration of drag.
- Default value:
releasingClass
— string- Default value:
"muuri-releasing"
. - This classname will be added to the item element for the duration of release.
- Default value:
Modify default settings
The default settings are stored in Muuri.defaultSettings
object.
Muuri.defaultSettings.positionDuration = 400;
Muuri.defaultSettings.dragSortAction = 'swap';
Quick reference
var defaults = {
// Container
container: null,
// Items
items: [],
positionDuration: 300,
positionEasing: 'ease-out',
show: {
duration: 300,
easing: 'ease-out'
},
hide: {
duration: 300,
easing: 'ease-out'
},
// Layout
layout: 'firstFit',
layoutOnResize: 100,
layoutOnInit: true,
// Drag & Drop
dragEnabled: false,
dragContainer: null,
dragPredicate: null,
dragSort: true,
dragSortInterval: 50,
dragSortTolerance: 50,
dragSortAction: 'move',
dragReleaseDuration: 300,
dragReleaseEasing: 'ease-out',
// Classnames
containerClass: 'muuri',
itemClass: 'muuri-item',
shownClass: 'muuri-shown',
hiddenClass: 'muuri-hidden',
positioningClass: 'muuri-positioning',
draggingClass: 'muuri-dragging',
releasingClass: 'muuri-releasing'
};
Bind an event on the Muuri instance.
Parameters
- event — string
- listener — function
Returns — object
Returns the instance.
Examples
muuri.on('layoutend', function (items) {
console.log(items);
});
Unbind an event from the Muuri instance.
Parameters
- event — string
- listener — function
Returns — object
Returns the instance.
Examples
var listener = function (items) {
console.log(items);
};
muuri
.on('layoutend', listener)
.off('layoutend', listener);
Recalculate the width and height of the provided targets. If no targets are provided all active items will be refreshed.
Parameters
- targets — array / element / Muuri.Item / number
- Optional.
- An array of DOM elements and/or
Muuri.Item
instances and/or integers (which describe the index of the item).
Examples
// Refresh all active items
muuri.refresh();
// Refresh the first item.
muuri.refresh(0);
// Refresh all items which match the provided DOM elements.
muuri.refresh([elemA, elemB]);
// Refresh specific items (instances of Muuri.Item).
muuri.refresh([itemA, itemB]);
Get all items. Optionally you can provide specific targets (indices or elements) and filter the results by the items' state (active/inactive). Note that the returned array is not the same object used by the instance so modifying it will not affect instance's items. All items that are not found are omitted from the returned array.
Parameters
- targets — array / element / Muuri.Item / number
- Optional.
- An array of DOM elements and/or
Muuri.Item
instances and/or integers (which describe the index of the item).
- state — string
- Optional.
- Default value:
undefined
. - Allowed values:
"active"
,"inactive"
. - Filter the returned array by the items' state. For example, if set to
"active"
all inactive items will be removed from the returned array.
Returns — array
Returns an array of Muuri.Item
instances.
Examples
// Get all items, active and inactive.
var allItems = muuri.get();
// Get all active items.
var activeItems = muuri.get('active');
// Get all inactive items.
var inactiveItems = muuri.get('inactive');
// Get the first item.
var firstItem = muuri.get(0)[0];
// Get specific items by their elements.
var items = muuri.get([elemA, elemB]);
// Get specific inactive items.
var items = muuri.get([elemA, elemB], 'inactive');
Add new items by providing the elements you wish to add to the instance and optionally provide the index where you want the items to be inserted into. All elements that are not already children of the container element will be automatically appended to the container.
If an element has it's CSS display property set to none it will be marked as inactive during the initiation process. As long as the item is inactive it will not be part of the layout, but it will retain it's index. You can activate items at any point with muuri.show()
method.
This method will automatically call muuri.layout()
if one or more of the added elements are visible. If only hidden items are added no layout will be called. All the new visible items are positioned without animation during their first layout.
Parameters
- elements — array / element
- An array of DOM elements.
- index — number
- Optional.
- Default value:
0
. - The index where you want the items to be inserted in. A value of
-1
will insert the items to the end of the list while0
will insert the items to the beginning. Note that the DOM elements are always just appended to the instance container regardless of the index value. You can use themuuri.synchronize()
method to arrange the DOM elments to the same order as the items.
Returns — array
Returns an array of Muuri.Item
instances.
Examples
// Add two new items to the beginning.
muuri.add([elemA, elemB]);
// Add two new items to the end.
muuri.add([elemA, elemB], -1);
Remove items from the instance.
Parameters
- targets — array / element / Muuri.Item / number
- An array of DOM elements and/or
Muuri.Item
instances and/or integers (which describe the index of the item).
- An array of DOM elements and/or
- removeElement — boolean
- Optional.
- Default value:
false
. - Should the associated DOM element be removed or not?
Returns — array
Returns the indices of the removed items.
Examples
// Remove the first item, but keep the element in the DOM.
muuri.remove(0);
// Remove items and the associated elements.
muuri.remove([elemA, elemB], true);
Order the item elements to match the order of the items. If the item's element is not a child of the container it is ignored and left untouched. This comes handy if you need to keep the DOM structure matched with the order of the items.
Examples
muuri.synchronize();
Calculate item positions and move items to their calculated positions unless they are already positioned correctly. The container's height is also adjusted according to the position of the items.
Parameters
- instant — boolean
- Optional.
- Default value:
false
. - Should the items be positioned instantly without any possible animation?
- callback — function
- Optional.
- A callback function that is called after the items have positioned. Receives two arguments. The first one is an array of all the items that were successfully positioned without interruptions and the second is a layout data object.
Examples
// Layout with animations (if any).
muuri.layout();
// Layout instantly without animations.
muuri.layout(true);
// Layout with callback (and with animations if any).
muuri.layout(function (items, layoutData) {
console.log('layout done!');
});
Show the targeted items.
Parameters
- targets — array / element / Muuri.Item / number
- An array of DOM elements and/or
Muuri.Item
instances and/or integers (which describe the index of the item).
- An array of DOM elements and/or
- instant — boolean
- Optional.
- Default value:
false
. - Should the items be shown instantly without any possible animation?
- callback — function
- Optional.
- A callback function that is called after the items are shown.
Examples
// Show items with animation (if any).
muuri.show([elemA, elemB]);
// Show items instantly without animations.
muuri.show([elemA, elemB], true);
// Show items with callback (and with animations if any).
muuri.show([elemA, elemB], function (items) {
console.log('items shown!');
});
Hide the targeted items.
Parameters
- targets — array / element / Muuri.Item / number
- An array of DOM elements and/or
Muuri.Item
instances and/or integers (which describe the index of the item).
- An array of DOM elements and/or
- instant — boolean
- Optional.
- Default value:
false
. - Should the items be hidden instantly without any possible animation?
- callback — function
- Optional.
- A callback function that is called after the items are hidden.
Examples
// Hide items with animation (if any).
muuri.hide([elemA, elemB]);
// Hide items instantly without animations.
muuri.hide([elemA, elemB], true);
// Hide items with callback (and with animations if any).
muuri.hide([elemA, elemB], function (items) {
console.log('items hidden!');
});
Get item's index.
Parameters
- target — element / Muuri.Item
Returns — number / null
Returns the target's index or null if the target is not found.
Examples
var index = muuri.indexOf(elemA);
Move item to another index or in place of another item.
Parameters
- targetFrom — element / Muuri.Item / number
- DOM element or
Muuri.Item
instance or index of the item as an integer.
- DOM element or
- targetTo — element / Muuri.Item / number
- DOM element or
Muuri.Item
instance or index of the item as an integer.
- DOM element or
Examples
// Move elemA to the index of elemB.
muuri.move(elemA, elemB);
// Move first item as last.
muuri.move(0, -1);
Swap positions of two items.
Parameters
- targetA — element / Muuri.Item / number
- DOM element or
Muuri.Item
instance or index of the item as an integer.
- DOM element or
- targetB — element / Muuri.Item / number
- DOM element or
Muuri.Item
instance or index of the item as an integer.
- DOM element or
Examples
// Swap positions of elemA and elemB.
muuri.swap(elemA, elemB);
// Swap positions of the first and the last item.
muuri.swap(0, -1);
Destroy the instance.
Examples
muuri.destroy();
Triggered after the muuri.refresh()
method is called.
Listener parameters
- items — array
- An array of
Muuri.Item
instances which were refreshed.
- An array of
Examples
muuri.on('refresh', function (items) {
console.log(items);
});
Triggered after the muuri.synchronize()
is called.
Examples
muuri.on('synchronize', function () {
console.log('Synced!');
});
Triggered when muuri.layout()
method is called, just before the items are positioned.
Listener parameters
- items — array
- An array of
Muuri.Item
instances that were succesfully positioned. If, for example, an item is being dragged it is ignored by the layout method.
- An array of
- layout — object
- A
Muuri.Layout
instance. - layout.muuri — Muuri
- A
Muuri
instance for which the layout was generated.
- A
- layout.items — array
- An array of
Muuri.Item
instances that were positioned.
- An array of
- layout.slots — object
- An object containing the positions of the
layout.items
. Indexed with the ids of the items. For example, to get the first item's position you would dolayout.slots[layout.items[0]._id]
. Each slot contains the the item's width, height, left and top.
- An object containing the positions of the
- layout.width — number
- The width of the grid.
- layout.height — number
- The height of the grid.
- A
Examples
muuri.on('layoutstart', function (items, layout) {
console.log(items);
console.log(layout);
});
Triggered when muuri.layout()
method is called, after the items have positioned.
Listener parameters
- items — array
- An array of
Muuri.Item
instances that were succesfully positioned. If, for example, an item is being dragged it is ignored by the layout method.
- An array of
- layout — object
- A
Muuri.Layout
instance. - layout.muuri — Muuri
- A
Muuri
instance for which the layout was generated.
- A
- layout.items — array
- An array of
Muuri.Item
instances that were positioned.
- An array of
- layout.slots — object
- An object containing the positions of the
layout.items
. Indexed with the ids of the items. For example, to get the first item's position you would dolayout.slots[layout.items[0]._id]
. Each slot contains the the item's width, height, left and top.
- An object containing the positions of the
- layout.width — number
- The width of the grid.
- layout.height — number
- The height of the grid.
- A
Examples
muuri.on('layoutend', function (items, layout) {
console.log(items);
console.log(layout);
});
Triggered when muuri.show()
method is called, just before the items are shown (with or without animation).
Listener parameters
- items — array
- An array of
Muuri.Item
instances that are about to be shown.
- An array of
Examples
muuri.on('showstart', function (items) {
console.log(items);
});
Triggered when muuri.show()
method is called, after the items are shown (with or without animation).
Listener parameters
- items — array
- An array of
Muuri.Item
instances that were succesfully shown without interruptions. If an item is already visible when themuuri.show()
method is called it is considered as successfully shown.
- An array of
Examples
muuri.on('showend', function (items) {
console.log(items);
});
Triggered when muuri.hide()
method is called, just before the items are hidden (with or without animation).
Listener parameters
- items — array
- An array of
Muuri.Item
instances that are about to be hidden.
- An array of
Examples
muuri.on('hidestart', function (items) {
console.log(items);
});
Triggered when muuri.hide()
method is called, after the items are hidden (with or without animation).
Listener parameters
- items — array
- An array of
Muuri.Item
instances that were succesfully hidden without interruptions. If an item is already hidden when themuuri.hide()
method is called it is considered as successfully hidden.
- An array of
Examples
muuri.on('hideend', function (items) {
console.log(items);
});
Triggered after muuri.move()
method is called.
Listener parameters
- targetFrom — Muuri.Item
Muuri.Item
instance that was moved.
- targetTo — Muuri.Item
Muuri.Item
instance to which's index the targetFrom item was moved to.
Examples
muuri.on('move', function (targetFrom, targetTo) {
console.log(targetFrom);
console.log(targetTo);
});
Triggered after muuri.swap()
method is called.
Listener parameters
- targetA — Muuri.Item
Muuri.Item
instance that was swapped with targetB.
- targetB — Muuri.Item
Muuri.Item
instance that was swapped with targetA.
Examples
muuri.on('swap', function (targetA, targetB) {
console.log(targetA);
console.log(targetB);
});
Triggered after muuri.add()
method is called.
Listener parameters
- items — array
- An array of
Muuri.Item
instances that were succesfully added to the muuri instance.
- An array of
Examples
muuri.on('add', function (items) {
console.log(items);
});
Triggered after muuri.remove()
method is called.
Listener parameters
- itemIndices — array
- Indices of the
Muuri.Item
instances that were succesfully removed from the muuri instance.
- Indices of the
Examples
muuri.on('remove', function (itemIndices) {
console.log(itemIndices);
});
Triggered when dragging of an item begins.
Listener parameters
- item — Muuri.Item
Muuri.Item
instance that is being dragged.
- data — object
- data.type — string
"dragstart"
- data.event — object
- The hammer event for the drag start event.
- data.currentLeft — number
- The dragged element's current translateX value.
- data.currentTop — object
- The dragged element's current translateY value.
- data.gridLeft — number
- The dragged element's current x-coordinate within the muuri container element.
- data.gridTop — object
- The dragged element's current y-coordinate within the muuri container element.
- data.type — string
Examples
muuri.on('dragstart', function (item, data) {
console.log(item);
console.log(data);
});
Triggered when an item is dragged.
Listener parameters
- item — Muuri.Item
Muuri.Item
instance that is being dragged.
- data — object
- data.type — string
"dragmove"
- data.event — object
- The hammer event for the drag start event.
- data.currentLeft — number
- The dragged element's current translateX value.
- data.currentTop — object
- The dragged element's current translateY value.
- data.gridLeft — number
- The dragged element's current x-coordinate within the muuri container element.
- data.gridTop — object
- The dragged element's current y-coordinate within the muuri container element.
- data.type — string
Examples
muuri.on('dragmove', function (item, data) {
console.log(item);
console.log(data);
});
Triggered when any of the scroll parents of a dragged item is scrolled.
Listener parameters
- item — Muuri.Item
Muuri.Item
instance that is being dragged.
- data — object
- data.type — string
"dragscroll"
- data.event — object
- Th scroll event.
- data.currentLeft — number
- The dragged element's current translateX value.
- data.currentTop — object
- The dragged element's current translateY value.
- data.gridLeft — number
- The dragged element's current x-coordinate within the muuri container element.
- data.gridTop — object
- The dragged element's current y-coordinate within the muuri container element.
- data.type — string
Examples
muuri.on('dragscroll', function (item, data) {
console.log(item);
console.log(data);
});
Triggered after item dragging ends.
Listener parameters
- item — Muuri.Item
Muuri.Item
instance that is being dragged.
- data — object
- data.type — string
"dragend"
- data.event — object
- The hammer event for the drag start event.
- data.currentLeft — number
- The dragged element's current translateX value.
- data.currentTop — object
- The dragged element's current translateY value.
- data.gridLeft — number
- The dragged element's current x-coordinate within the muuri container element.
- data.gridTop — object
- The dragged element's current y-coordinate within the muuri container element.
- data.type — string
Examples
muuri.on('dragend', function (item, data) {
console.log(item);
console.log(data);
});
Triggered when item is released (right after dragging ends).
Listener parameters
- item — Muuri.Item
Muuri.Item
instance that is being dragged.
Examples
muuri.on('releasestart', function (item) {
console.log(item);
});
Triggered after item has finished the release procedure (animated back to it's position if animations are enabled).
Listener parameters
- item — Muuri.Item
Muuri.Item
instance that is being dragged.
Examples
muuri.on('releaseend', function (item) {
console.log(item);
});
Triggered after muuri.destroy()
method is called.
Examples
muuri.on('destroy', function () {
console.log('Muuri is no more...');
});
Copyright © 2015 Haltu Oy. Licensed under the MIT license.