记录拖动排序
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
<script src="https://cdn.staticfile.org/jquery/1.10.2/jquery.min.js"></script>
<script src="//code.jquery.com/ui/1.10.4/jquery-ui.js"></script>
<script src="https://cdn.staticfile.org/angular.js/1.6.6/angular.min.js"></script>
<script src="./sortable.js"></script>
</head>
<body ng-app="app">
<div ng-controller="sortCtrl">
<ul ui-sortable="sortableOptions" ng-model="data">
<li ng-repeat="item in data ">
<span>{{item.name}}, {{item.age}}</span>
</li>
</ul>
</div>
</body> <script>
angular.module("app", ["ui.sortable"])
.controller("sortCtrl", function($scope, $timeout) {
$scope.cannotSort = false;
$scope.data = [{
"name": "allen",
"age": 21,
"i": 0
}, {
"name": "bob",
"age": 18,
"i": 1
}, {
"name": "curry",
"age": 25,
"i": 2
}, {
"name": "david",
"age": 30,
"i": 3
}]; $scope.sortableOptions = {
// 数据有变化
update: function(e, ui) {
console.log("update");
//需要使用延时方法,否则会输出原始数据的顺序,可能是BUG?
$timeout(function() {
var resArr = [];
for (var i = 0; i < $scope.data.length; i++) {
resArr.push($scope.data[i].i);
}
console.log(resArr);
}) }, // 完成拖拽动作
stop: function(e, ui) {
//do nothing
console.log("do nothing");
}
}
})
</script> </html>
sortable.js
/*
jQuery UI Sortable plugin wrapper @param [ui-sortable] {object} Options to pass to $.fn.sortable() merged onto ui.config
*/
angular
.module('ui.sortable', [])
.value('uiSortableConfig', {
// the default for jquery-ui sortable is "> *", we need to restrict this to
// ng-repeat items
// if the user uses
items: '> [ng-repeat],> [data-ng-repeat],> [x-ng-repeat]'
})
.directive('uiSortable', [
'uiSortableConfig',
'$timeout',
'$log',
function(uiSortableConfig, $timeout, $log) {
return {
require: '?ngModel',
scope: {
ngModel: '=',
uiSortable: '=',
////Expression bindings from html.
create: '&uiSortableCreate',
// helper:'&uiSortableHelper',
start: '&uiSortableStart',
activate: '&uiSortableActivate',
// sort:'&uiSortableSort',
// change:'&uiSortableChange',
// over:'&uiSortableOver',
// out:'&uiSortableOut',
beforeStop: '&uiSortableBeforeStop',
update: '&uiSortableUpdate',
remove: '&uiSortableRemove',
receive: '&uiSortableReceive',
deactivate: '&uiSortableDeactivate',
stop: '&uiSortableStop'
},
link: function(scope, element, attrs, ngModel) {
var savedNodes;
var helper; function combineCallbacks(first, second) {
var firstIsFunc = typeof first === 'function';
var secondIsFunc = typeof second === 'function';
if (firstIsFunc && secondIsFunc) {
return function() {
first.apply(this, arguments);
second.apply(this, arguments);
};
} else if (secondIsFunc) {
return second;
}
return first;
} function getSortableWidgetInstance(element) {
// this is a fix to support jquery-ui prior to v1.11.x
// otherwise we should be using `element.sortable('instance')`
var data = element.data('ui-sortable');
if (
data &&
typeof data === 'object' &&
data.widgetFullName === 'ui-sortable'
) {
return data;
}
return null;
} function setItemChildrenWidth(item) {
item.children().each(function() {
var $el = angular.element(this); // Preserve the with of the element
$el.width($el.width());
});
} function dummyHelper(e, item) {
return item;
} function patchSortableOption(key, value) {
if (callbacks[key]) {
if (key === 'stop') {
// call apply after stop
value = combineCallbacks(value, function() {
scope.$apply();
}); value = combineCallbacks(value, afterStop);
}
// wrap the callback
value = combineCallbacks(callbacks[key], value);
} else if (wrappers[key]) {
value = wrappers[key](value);
} // patch the options that need to have values set
if (!value && (key === 'items' || key === 'ui-model-items')) {
value = uiSortableConfig.items;
} return value;
} function patchUISortableOptions(
newOpts,
oldOpts,
sortableWidgetInstance
) {
function addDummyOptionKey(value, key) {
if (!(key in opts)) {
// add the key in the opts object so that
// the patch function detects and handles it
opts[key] = null;
}
}
// for this directive to work we have to attach some callbacks
angular.forEach(callbacks, addDummyOptionKey); // only initialize it in case we have to
// update some options of the sortable
var optsDiff = null; if (oldOpts) {
// reset deleted options to default
var defaultOptions;
angular.forEach(oldOpts, function(oldValue, key) {
if (!newOpts || !(key in newOpts)) {
if (key in directiveOpts) {
if (key === 'ui-floating') {
opts[key] = 'auto';
} else {
opts[key] = patchSortableOption(key, undefined);
}
return;
} if (!defaultOptions) {
defaultOptions = angular.element.ui.sortable().options;
}
var defaultValue = defaultOptions[key];
defaultValue = patchSortableOption(key, defaultValue); if (!optsDiff) {
optsDiff = {};
}
optsDiff[key] = defaultValue;
opts[key] = defaultValue;
}
});
} newOpts = angular.extend({}, newOpts);
// update changed options
// handle the custom option of the directive first
angular.forEach(newOpts, function(value, key) {
if (key in directiveOpts) {
if (
key === 'ui-floating' &&
(value === false || value === true) &&
sortableWidgetInstance
) {
sortableWidgetInstance.floating = value;
} if (
key === 'ui-preserve-size' &&
(value === false || value === true)
) {
var userProvidedHelper = opts.helper;
newOpts.helper = function(e, item) {
if (opts['ui-preserve-size'] === true) {
setItemChildrenWidth(item);
}
return (userProvidedHelper || dummyHelper).apply(
this,
arguments
);
};
} opts[key] = patchSortableOption(key, value);
}
}); // handle the normal option of the directive
angular.forEach(newOpts, function(value, key) {
if (key in directiveOpts) {
// the custom option of the directive are already handled
return;
} value = patchSortableOption(key, value); if (!optsDiff) {
optsDiff = {};
}
optsDiff[key] = value;
opts[key] = value;
}); return optsDiff;
} function getPlaceholderElement(element) {
var placeholder = element.sortable('option', 'placeholder'); // placeholder.element will be a function if the placeholder, has
// been created (placeholder will be an object). If it hasn't
// been created, either placeholder will be false if no
// placeholder class was given or placeholder.element will be
// undefined if a class was given (placeholder will be a string)
if (
placeholder &&
placeholder.element &&
typeof placeholder.element === 'function'
) {
var result = placeholder.element();
// workaround for jquery ui 1.9.x,
// not returning jquery collection
result = angular.element(result);
return result;
}
return null;
} function getPlaceholderExcludesludes(element, placeholder) {
// exact match with the placeholder's class attribute to handle
// the case that multiple connected sortables exist and
// the placeholder option equals the class of sortable items
var notCssSelector = opts['ui-model-items'].replace(/[^,]*>/g, '');
var excludes = element.find(
'[class="' +
placeholder.attr('class') +
'"]:not(' +
notCssSelector +
')'
);
return excludes;
} function hasSortingHelper(element, ui) {
var helperOption = element.sortable('option', 'helper');
return (
helperOption === 'clone' ||
(typeof helperOption === 'function' &&
ui.item.sortable.isCustomHelperUsed())
);
} function getSortingHelper(element, ui /*, savedNodes*/) {
var result = null;
if (
hasSortingHelper(element, ui) &&
element.sortable('option', 'appendTo') === 'parent'
) {
// The .ui-sortable-helper element (that's the default class name)
result = helper;
}
return result;
} // thanks jquery-ui
function isFloating(item) {
return (
/left|right/.test(item.css('float')) ||
/inline|table-cell/.test(item.css('display'))
);
} function getElementContext(elementScopes, element) {
for (var i = 0; i < elementScopes.length; i++) {
var c = elementScopes[i];
if (c.element[0] === element[0]) {
return c;
}
}
} function afterStop(e, ui) {
ui.item.sortable._destroy();
} // return the index of ui.item among the items
// we can't just do ui.item.index() because there it might have siblings
// which are not items
function getItemIndex(item) {
return item
.parent()
.find(opts['ui-model-items'])
.index(item);
} var opts = {}; // directive specific options
var directiveOpts = {
'ui-floating': undefined,
'ui-model-items': uiSortableConfig.items,
'ui-preserve-size': undefined
}; var callbacks = {
create: null,
start: null,
activate: null,
// sort: null,
// change: null,
// over: null,
// out: null,
beforeStop: null,
update: null,
remove: null,
receive: null,
deactivate: null,
stop: null
}; var wrappers = {
helper: null
}; angular.extend(
opts,
directiveOpts,
uiSortableConfig,
scope.uiSortable
); if (!angular.element.fn || !angular.element.fn.jquery) {
$log.error(
'ui.sortable: jQuery should be included before AngularJS!'
);
return;
} function wireUp() {
// When we add or remove elements, we need the sortable to 'refresh'
// so it can find the new/removed elements.
scope.$watchCollection('ngModel', function() {
// Timeout to let ng-repeat modify the DOM
$timeout(
function() {
// ensure that the jquery-ui-sortable widget instance
// is still bound to the directive's element
if (!!getSortableWidgetInstance(element)) {
element.sortable('refresh');
}
},
0,
false
);
}); callbacks.start = function(e, ui) {
if (opts['ui-floating'] === 'auto') {
// since the drag has started, the element will be
// absolutely positioned, so we check its siblings
var siblings = ui.item.siblings();
var sortableWidgetInstance = getSortableWidgetInstance(
angular.element(e.target)
);
sortableWidgetInstance.floating = isFloating(siblings);
} // Save the starting position of dragged item
var index = getItemIndex(ui.item);
ui.item.sortable = {
model: ngModel.$modelValue[index],
index: index,
source: element,
sourceList: ui.item.parent(),
sourceModel: ngModel.$modelValue,
cancel: function() {
ui.item.sortable._isCanceled = true;
},
isCanceled: function() {
return ui.item.sortable._isCanceled;
},
isCustomHelperUsed: function() {
return !!ui.item.sortable._isCustomHelperUsed;
},
_isCanceled: false,
_isCustomHelperUsed: ui.item.sortable._isCustomHelperUsed,
_destroy: function() {
angular.forEach(ui.item.sortable, function(value, key) {
ui.item.sortable[key] = undefined;
});
},
_connectedSortables: [],
_getElementContext: function(element) {
return getElementContext(this._connectedSortables, element);
}
};
}; callbacks.activate = function(e, ui) {
var isSourceContext = ui.item.sortable.source === element;
var savedNodesOrigin = isSourceContext
? ui.item.sortable.sourceList
: element;
var elementContext = {
element: element,
scope: scope,
isSourceContext: isSourceContext,
savedNodesOrigin: savedNodesOrigin
};
// save the directive's scope so that it is accessible from ui.item.sortable
ui.item.sortable._connectedSortables.push(elementContext); // We need to make a copy of the current element's contents so
// we can restore it after sortable has messed it up.
// This is inside activate (instead of start) in order to save
// both lists when dragging between connected lists.
savedNodes = savedNodesOrigin.contents();
helper = ui.helper; // If this list has a placeholder (the connected lists won't),
// don't inlcude it in saved nodes.
var placeholder = getPlaceholderElement(element);
if (placeholder && placeholder.length) {
var excludes = getPlaceholderExcludesludes(
element,
placeholder
);
savedNodes = savedNodes.not(excludes);
}
}; callbacks.update = function(e, ui) {
// Save current drop position but only if this is not a second
// update that happens when moving between lists because then
// the value will be overwritten with the old value
if (!ui.item.sortable.received) {
ui.item.sortable.dropindex = getItemIndex(ui.item);
var droptarget = ui.item
.parent()
.closest(
'[ui-sortable], [data-ui-sortable], [x-ui-sortable]'
);
ui.item.sortable.droptarget = droptarget;
ui.item.sortable.droptargetList = ui.item.parent(); var droptargetContext = ui.item.sortable._getElementContext(
droptarget
);
ui.item.sortable.droptargetModel =
droptargetContext.scope.ngModel; // Cancel the sort (let ng-repeat do the sort for us)
// Don't cancel if this is the received list because it has
// already been canceled in the other list, and trying to cancel
// here will mess up the DOM.
element.sortable('cancel');
} // Put the nodes back exactly the way they started (this is very
// important because ng-repeat uses comment elements to delineate
// the start and stop of repeat sections and sortable doesn't
// respect their order (even if we cancel, the order of the
// comments are still messed up).
var sortingHelper =
!ui.item.sortable.received &&
getSortingHelper(element, ui, savedNodes);
if (sortingHelper && sortingHelper.length) {
// Restore all the savedNodes except from the sorting helper element.
// That way it will be garbage collected.
savedNodes = savedNodes.not(sortingHelper);
}
var elementContext = ui.item.sortable._getElementContext(element);
savedNodes.appendTo(elementContext.savedNodesOrigin); // If this is the target connected list then
// it's safe to clear the restored nodes since:
// update is currently running and
// stop is not called for the target list.
if (ui.item.sortable.received) {
savedNodes = null;
} // If received is true (an item was dropped in from another list)
// then we add the new item to this list otherwise wait until the
// stop event where we will know if it was a sort or item was
// moved here from another list
if (ui.item.sortable.received && !ui.item.sortable.isCanceled()) {
scope.$apply(function() {
ngModel.$modelValue.splice(
ui.item.sortable.dropindex,
0,
ui.item.sortable.moved
);
});
scope.$emit('ui-sortable:moved', ui);
}
}; callbacks.stop = function(e, ui) {
// If the received flag hasn't be set on the item, this is a
// normal sort, if dropindex is set, the item was moved, so move
// the items in the list.
var wasMoved =
'dropindex' in ui.item.sortable &&
!ui.item.sortable.isCanceled(); if (wasMoved && !ui.item.sortable.received) {
scope.$apply(function() {
ngModel.$modelValue.splice(
ui.item.sortable.dropindex,
0,
ngModel.$modelValue.splice(ui.item.sortable.index, 1)[0]
);
});
scope.$emit('ui-sortable:moved', ui);
} else if (
!wasMoved &&
!angular.equals(
element.contents().toArray(),
savedNodes.toArray()
)
) {
// if the item was not moved
// and the DOM element order has changed,
// then restore the elements
// so that the ngRepeat's comment are correct. var sortingHelper = getSortingHelper(element, ui, savedNodes);
if (sortingHelper && sortingHelper.length) {
// Restore all the savedNodes except from the sorting helper element.
// That way it will be garbage collected.
savedNodes = savedNodes.not(sortingHelper);
}
var elementContext = ui.item.sortable._getElementContext(
element
);
savedNodes.appendTo(elementContext.savedNodesOrigin);
} // It's now safe to clear the savedNodes and helper
// since stop is the last callback.
savedNodes = null;
helper = null;
}; callbacks.receive = function(e, ui) {
// An item was dropped here from another list, set a flag on the
// item.
ui.item.sortable.received = true;
}; callbacks.remove = function(e, ui) {
// Workaround for a problem observed in nested connected lists.
// There should be an 'update' event before 'remove' when moving
// elements. If the event did not fire, cancel sorting.
if (!('dropindex' in ui.item.sortable)) {
element.sortable('cancel');
ui.item.sortable.cancel();
} // Remove the item from this list's model and copy data into item,
// so the next list can retrive it
if (!ui.item.sortable.isCanceled()) {
scope.$apply(function() {
ui.item.sortable.moved = ngModel.$modelValue.splice(
ui.item.sortable.index,
1
)[0];
});
}
}; // setup attribute handlers
angular.forEach(callbacks, function(value, key) {
callbacks[key] = combineCallbacks(callbacks[key], function() {
var attrHandler = scope[key];
var attrHandlerFn;
if (
typeof attrHandler === 'function' &&
(
'uiSortable' +
key.substring(0, 1).toUpperCase() +
key.substring(1)
).length &&
typeof (attrHandlerFn = attrHandler()) === 'function'
) {
attrHandlerFn.apply(this, arguments);
}
});
}); wrappers.helper = function(inner) {
if (inner && typeof inner === 'function') {
return function(e, item) {
var oldItemSortable = item.sortable;
var index = getItemIndex(item); item.sortable = {
model: ngModel.$modelValue[index],
index: index,
source: element,
sourceList: item.parent(),
sourceModel: ngModel.$modelValue,
_restore: function() {
angular.forEach(item.sortable, function(value, key) {
item.sortable[key] = undefined;
}); item.sortable = oldItemSortable;
}
}; var innerResult = inner.apply(this, arguments);
item.sortable._restore();
item.sortable._isCustomHelperUsed = item !== innerResult;
return innerResult;
};
}
return inner;
}; scope.$watchCollection(
'uiSortable',
function(newOpts, oldOpts) {
// ensure that the jquery-ui-sortable widget instance
// is still bound to the directive's element
var sortableWidgetInstance = getSortableWidgetInstance(element);
if (!!sortableWidgetInstance) {
var optsDiff = patchUISortableOptions(
newOpts,
oldOpts,
sortableWidgetInstance
); if (optsDiff) {
element.sortable('option', optsDiff);
}
}
},
true
); patchUISortableOptions(opts);
} function init() {
if (ngModel) {
wireUp();
} else {
$log.info('ui.sortable: ngModel not provided!', element);
} // Create sortable
element.sortable(opts);
} function initIfEnabled() {
if (scope.uiSortable && scope.uiSortable.disabled) {
return false;
} init(); // Stop Watcher
initIfEnabled.cancelWatcher();
initIfEnabled.cancelWatcher = angular.noop; return true;
} initIfEnabled.cancelWatcher = angular.noop; if (!initIfEnabled()) {
initIfEnabled.cancelWatcher = scope.$watch(
'uiSortable.disabled',
initIfEnabled
);
}
}
};
}
]);