Lightweight vanillajs micro-library for creating sortable lists and grids using native HTML5 drag and drop API.
- Only 2KB (minified and gzipped).
- Built using native HTML5 drag and drop API. No dependencies.
- Supports both list and grid style layouts.
- Supported Browsers: Current versions of all major browsers (Chrome, Firefox, Safari, Opera, Edge), IE11+
- Supports exports as AMD, CommonJS or global
Demo: Check out the examples
If you would like to add an adapter to the list, please create an issue with the link to your adapter.
You need to install the package using npm
or downloading it manually. Afterwards you need to load dist/html.sortable.js
or the minified version, dist/html.sortable.min.js
. Make sure to grab the file from the dist/
directory.
npm install html5sortable --save
Still using bower? Look here.
You can find the examples online or test locally. Warning: the online demo is just to show off the features and is most likely not up to date. Please study this readme file for the current way of implementing and using html5sortable
.
Use sortable
method to create a sortable list:
sortable('.sortable');
Use .sortable-placeholder
CSS selectors to change the styles of the placeholder. You may change the class by setting the placeholderClass
option in the config object.
sortable('.sortable', {
placeholderClass: 'my-placeholder fade'
});
You can nest sortables inside each other. However, take care to add a wrapper around the items, a sortable-item can not at the same time be a sortable
.
<div class="list"><!-- Sortable -->
<div class="item"> Item 1
<div class="sublist"><!-- Nested Sortable; Wrapping container needed -->
<div class="subitem">Subitem 1</div>
<div class="subitem">Subitem 2</div>
</div>
</div>
<div class="item"> Item 2 </div>
</div>
NOTE: Events can be listened on any element from the group (when using connectWith
), since the same event will be dispatched on all of them.
Use sortstart
event if you want to do something when sorting starts:
sortable('.sortable')[0].addEventListener('sortstart', function(e) {
/*
This event is triggered when the user starts sorting and the DOM position has not yet changed.
e.detail.item contains the current dragged element
e.detail.placeholder contains the placeholder element
e.detail.startparent contains the element that the dragged item comes from
*/
});
Use the sortstop
event if you want to do something when sorting stops:
sortable('.sortable')[0].addEventListener('sortstop', function(e) {
/*
This event is triggered when the user stops sorting. The DOM position may have changed.
e.detail.item contains the element that was dragged.
e.detail.startparent contains the element that the dragged item came from.
*/
});
Use sortupdate
event if you want to do something when the order changes (e.g. storing the new order):
sortable('.sortable')[0].addEventListener('sortupdate', function(e) {
/*
This event is triggered when the user stopped sorting and the DOM position has changed.
e.detail.item contains the current dragged element.
e.detail.index contains the new index of the dragged element (considering only list items)
e.detail.oldindex contains the old index of the dragged element (considering only list items)
e.detail.elementIndex contains the new index of the dragged element (considering all items within sortable)
e.detail.oldElementIndex contains the old index of the dragged element (considering all items within sortable)
e.detail.startparent contains the element that the dragged item comes from
e.detail.endparent contains the element that the dragged item was added to (new parent)
e.detail.newEndList contains all elements in the list the dragged item was dragged to
e.detail.newStartList contains all elements in the list the dragged item was dragged from
e.detail.oldStartList contains all elements in the list the dragged item was dragged from BEFORE it was dragged from it
*/
});
Use the items
option to specify which items inside the element should be sortable:
sortable('.sortable', {
items: ':not(.disabled)'
});
Use the handle
option to restrict drag start to the specified element:
sortable('.sortable', {
handle: 'h2'
});
Setting the forcePlaceholderSize
option to true, forces the placeholder to have a height:
sortable('.sortable', {
forcePlaceholderSize: true
});
Use the connectWith
option to create a connected lists:
sortable('.js-sortable, .js-second-sortable', {
connectWith: 'connected' // unique string, which is not used for other connectWith sortables
});
Use the placeholder
option to specify the markup of the placeholder:
sortable('.sortable', {
items: 'tr' ,
placeholder: '<tr><td colspan="7"> </td></tr>'
});
Use the hoverClass
option to specify applying a css class to the hovered element rather than relying on :hover
. This can eliminate some potential drag and drop issues where another element thinks it is being hovered over.
sortable('.sortable', {
hoverClass: 'is-hovered' // Defaults to false
});
Use the maxItems
option to restrict the number of items that can be added to a sortable from a connected sortable. maxItems
should always be combined with the items
option. Make sure items
does not match placeholder and other options, so that they are not counted.
sortable('.sortable', {
maxItems: 3 // Defaults to 0 (no limit)
});
To remove the sortable functionality completely:
sortable('.sortable', 'destroy');
To disable the sortable temporarily:
sortable('.sortable', 'disable');
To enable a disabled sortable:
sortable('.sortable', 'enable');
To serialize a sortable:
sortable('.sortable', 'serialize');
This will return an array of objects, each with a list
key for the sortable and a children
key for the children.
[
0: {
list: ul.js-sortable // Object
children: [
0: li, // object
1: li // object
]
}
]
When you add a new item to a sortable, it will not automatically be a draggable item, so you will need to reinit the sortable. Your previously added options will be preserved.
sortable('.sortable');
The plugin has limited support for sorting table rows. To sort table rows:
- Initialize plugin on
tbody
element - Keep in mind that different browsers may display different ghost image of the row during the drag action. Webkit browsers seem to hide entire contents of
td
cell if there are any inline elements inside thetd
. This may or may not be fixed by setting thetd
to beposition: relative;
This version is maintained by Lukas Oppermann and many other contributors. Thanks for your help! 👍
Contributions are always welcome. Please check out the contribution guidelines to make it fast & easy for us to merge your PR.
- Dragstart not working on buttons
Dragstart event does not fire onbutton
elements. This effectively disables drag and drop for button elements. See https://caniuse.com/#feat=dragndrop in the known issues section.