JavaScript实现拖放效果
JavaScript实现拖放效果
笔者实现该效果也是套用别人的轮子的。传送门
然后厚颜无耻的贴别人的readme~,笔者为了方便查阅就直接贴了,有不想移步的可以看这篇。不过还是最好请到原作者的GitHub去查看,支持一下原作者。
文章最后有贴dnd.js
的源码。
使用介绍
dnd.js
拖放库 drag and drop
不依赖任何第三方库的拖放库,兼容低版本浏览器,兼容移动端,自带常用动画效果,如果在vue项目中使用,可以拷贝上面vue文件夹中已经封装好的组件直接使用.
安装方法
- 方式一
npm install dnd.js --save
- 方式二
下载项目中的dist/dnd.js, 然后用script标签插入到你的项目中, 如下
这种方式可以通过window.dnd访问
<script type="text/javascript" src="dist/dnd.js"></script>
使用方法
- 最简单的使用方法可以可以查看最小demo
- 也可以查看稍微复杂一点的demo
import { Drag, Drop } from 'dnd.js'
new Drag(element, options)
new Drop(element, options)
- 如果是使用方式二引入的话, 可以这样使用
var Drag = dnd.Drag
var Drop = dnd.Drop
new Drag(element, options)
new Drop(element, options)
- 最小demo代码展示
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>最小化demo</title>
<script type="text/javascript" src="../dist/dnd.js"></script>
<style>
html, body { width:100%; height:100%; }
body { margin:0; padding:20px; box-sizing:border-box; }
.drop { margin:20px auto; width:100%; height:100px; border:1px solid #000; }
</style>
</head>
<body>
<div class="drop"></div>
<div class="drag">
拖动我到上方框框
</div>
</body>
<script>
var Drag = dnd.Drag
var Drop = dnd.Drop
new Drag('.drag')
new Drop('.drop', {
onDrop: function (params) {
params.el.appendChild(params.sourceNode)
}
})
</script>
</html>
参数说明
- 创建
new Drag(element, options)
new Drop(element, options)
参数 | 是否必填 | 类型 | 说明 |
---|---|---|---|
element | 是 | String or htmlElementObject | 可以传入类名class 或者 id名 或者 dom节点 |
options | 是 | Object | 相关参数和回调函数,具体说明看下面 |
- options说明 (Drag)
例如:
let element = '.drag-container'
let options = {
data: '本次拖动希望传递给Drop对象的参数',
onDragStart: function (params) {
console.log('监听到拖动开始')
},
onDragEnd: function (params) {
console.log('监听到拖动结束')
}
}
new Drag(element, options)
options属性说明:
参数 | 是否必填 | 类型 | 说明 |
---|---|---|---|
data | 否 | 任意 | 本次拖动希望传递给Drop对象的参数 |
onDragStart | 否 | Function | 拖动开始回调函数 |
onDragEnd | 否 | Function | 拖动结束回调函数 |
- options说明 (Drop)
例如:
let element = '.drop-container'
let options = {
name: '当前Drop对象的名字',
onDragStart (params) {
console.log('监听到拖动开始')
},
onDragEnter (params) {
console.log('监听到被拖元素进入')
},
onDragOver (params) {
console.log('监听到被拖动元素在自己上方移动')
console.log('这个函数会被连续调用')
},
onDragLeave (params) {
console.log('监听被拖动元素离开')
},
onDrop (params) {
console.log('监听到被拖动元素在自己上方放下')
},
onDragEnd (params) {
console.log('监听到拖动结束')
}
}
new Drop(element, options)
options属性说明:
属性 | 是否必填 | 类型 | 说明 | 回调函数参数说明 |
---|---|---|---|---|
name | 否 | String | 定义当前Drop对象的名字 | - |
onDragStart | 否 | Function | 拖动开始时调用 | 见下方说明 |
onDragEnter | 否 | Function | 拖动进入时调用 | 见下方说明 |
onDragOver | 否 | Function | 被拖动元素在自己上方移动时候调用, 这个函数会被连续调用 | 见下方说明 |
onDragLeave | 否 | Function | 拖动离开时候调用 | 见下方说明 |
onDrop | 是 | Function | 被拖动元素在自己上方放下时调用 | 见下方说明 |
onDragEnd | 否 | Function | 拖动结束时候调用 | 见下方说明 |
回调函数的参数params说明:
属性 | 类型 | 描述 |
---|---|---|
data | 不定 | 被拖动元素定义的data属性, 类型由Drag对象被创建的时候传入的data属性决定 |
el | Object | 当前dom节点 |
enter | Boolean | 是否进入当前范围的标志位, 布尔值 |
methods | Object | 见下方 回调函数参数中的methods说明 |
name | String | Drop名称, 在销毁当前Drag对象时候需要用到 |
sourceNode | Object | 被拖动元素的dom节点 |
回调函数的参数中的methods对象说明:
提供一些方法供回调函数调用
showStateIcon: 显示状态图标
例如:showStateIcon('add')
showStateIcon('error')
你也可以传入自己喜欢的图片 例如这样:
showStateIcon('https://ss2.bdstatic.com/lfoZeXSm1A5BphGlnYG/icon/10062.png')
还有 showStateIcon('delete') 和 showStateIcon('reject') 分别出现'删除'图标和'禁止'的图标hideStateIcon: 隐藏状态图标
removeDragedNode: 移除跟随鼠标移动的被拖元素
例如:removeDragedNode('fade')
removeDragedNode('back')
removeDragedNode('blost')
getStateIconNode: 获取跟随鼠标移动的状态图标dom节点
destroyDrop: 销毁当前Drop对象
方法名 | 示例 | 参数说明 | 描述 |
---|---|---|---|
showStateIcon | params.methods.showStateIcon('add') | 参数类型: String, 内置三种常用图标: 添加('add'), 错误('error'), 删除('delete'), 拒绝(reject),传入对应的名字即可, 也可以自定义图标,直接传入图片的完整地址 | 显示状态图标, 调用后会在鼠标旁边出现一个跟随鼠标移动的小图标, 如果要隐藏只需要调用hideStateIcon函数即可 |
hideStateIcon | params.methods.hideStateIcon() | 无参数 | 隐藏跟随鼠标移动的状态图标(如果没有调用showStateIcon函数的话图标默认不显示) |
removeDragedNode | params.methods.removeDragedNode(animationName, time) | animationName(动画类型), 参数类型:String, 非必填, 可选: fade / blost / back, time(动画时间, 单位毫秒), 参数类型: Number, 非必填 | 移除跟随鼠标移动的被拖元素, 如果没有参数则直接消失, 有参数表示执行消失动画后再消失 目前支持三种动画, 分别是: 褪色(fade), 爆炸(blost), 反弹(back), 三种动画对应不同也应用场景, 例如: removeDragedNode('blost', 300) |
getStateIconNode | params.methods.getStateIconNode() | 无参数 | 返回跟随鼠标移动的状态图标dom节点 |
destroyDrop | params.methods.destroyDrop({name}) | 参数类型: String, Drag对象的名字 | 销毁Drop对象,匹配所有名字跟传入参数一致的Drop对象并销毁, 销毁后将不能接收Drag对象 |
dnd.js源码
(function webpackUniversalModuleDefinition(root, factory) {
if(typeof exports === 'object' && typeof module === 'object')
module.exports = factory();
else if(typeof define === 'function' && define.amd)
define("dnd", [], factory);
else if(typeof exports === 'object')
exports["dnd"] = factory();
else
root["dnd"] = factory();
})(this, function() {
return /******/ (function(modules) { // webpackBootstrap
/******/ // The module cache
/******/ var installedModules = {};
/******/
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/
/******/ // Check if module is in cache
/******/ if(installedModules[moduleId])
/******/ return installedModules[moduleId].exports;
/******/
/******/ // Create a new module (and put it into the cache)
/******/ var module = installedModules[moduleId] = {
/******/ i: moduleId,
/******/ l: false,
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/
/******/ // Flag the module as loaded
/******/ module.l = true;
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/
/******/
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = modules;
/******/
/******/ // expose the module cache
/******/ __webpack_require__.c = installedModules;
/******/
/******/ // identity function for calling harmony imports with the correct context
/******/ __webpack_require__.i = function(value) { return value; };
/******/
/******/ // define getter function for harmony exports
/******/ __webpack_require__.d = function(exports, name, getter) {
/******/ if(!__webpack_require__.o(exports, name)) {
/******/ Object.defineProperty(exports, name, {
/******/ configurable: false,
/******/ enumerable: true,
/******/ get: getter
/******/ });
/******/ }
/******/ };
/******/
/******/ // getDefaultExport function for compatibility with non-harmony modules
/******/ __webpack_require__.n = function(module) {
/******/ var getter = module && module.__esModule ?
/******/ function getDefault() { return module['default']; } :
/******/ function getModuleExports() { return module; };
/******/ __webpack_require__.d(getter, 'a', getter);
/******/ return getter;
/******/ };
/******/
/******/ // Object.prototype.hasOwnProperty.call
/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
/******/
/******/ // __webpack_public_path__
/******/ __webpack_require__.p = "";
/******/
/******/ // Load entry module and return exports
/******/ return __webpack_require__(__webpack_require__.s = 7);
/******/ })
/************************************************************************/
/******/ ([
/* 0 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
var DOCUMENT_ADDR = exports.DOCUMENT_ADDR = 'https://github.com/qgh810/dnd';
var REMOVE_ANIMATION_TYPES = exports.REMOVE_ANIMATION_TYPES = {
'fade': 'animation_fade',
'blost': 'animation_blost',
'back': 'animation_back'
};
/***/ }),
/* 1 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.methods = undefined;
var _dragStore;
var _iconImages = __webpack_require__(5);
var _iconImages2 = _interopRequireDefault(_iconImages);
var _config = __webpack_require__(0);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
var dragStore = (_dragStore = {
/* ********** 被拖元素drag设置的变量 *************/
//
isMobile: false,
data: null,
draggedNode: null,
sourceNode: null,
markNode: null,
stateIcon: null,
mousePosition: null,
/* ********** drop设置的变量 *************/
targets: [],
targetOnDragStarts: [],
targetOnDragEnds: [],
onDragEnters: [],
onDragLeaves: [],
onDragOvers: [],
onDrops: [],
targetPositions: [],
/* ********** store设置的变量 *************/
targetIndex: -1,
_prevValidIndex: -1,
_inTarget: false,
/**
* 监听拖动开始
*/
onDragStart: function onDragStart(data, el) {
var _this = this;
this._initStore();
this.sourceNode = el;
this.data = data;
// 广播拖动开始
this.targetOnDragStarts.forEach(function (fn, index) {
return fn && fn({
data: data,
enter: index === _this.targetIndex,
el: _this.targets[index].el,
sourceNode: _this.sourceNode,
name: _this.targets[index].name,
expand: _this.targets[index].expand,
methods: methods
});
});
},
/**
* 初始化store状态
*/
_initStore: function _initStore() {
this._inTarget = false;
this._prevValidIndex = -1;
this.targetIndex = -1;
this.hideStateIcon();
},
/**
* 监听拖动
*/
onDragOver: function onDragOver(pageX, pageY) {
this.mousePosition = [pageX, pageY];
this.targetIndex = this.collision(pageX, pageY);
this.setIconPosition(pageX, pageY);
if (this.targetIndex >= 0) {
// 判断是否在目标外 是的话表示刚刚进入
if (!this._inTarget) {
this._prevValidIndex = this.targetIndex;
this._inTarget = true;
var _params = {
enter: true,
data: this.data,
el: this.targets[this.targetIndex].el,
sourceNode: this.sourceNode,
name: this.targets[this.targetIndex].name,
expand: this.targets[this.targetIndex].expand,
methods: methods
};
this.onDragEnters[this.targetIndex](_params);
}
// 调用回调
var params = {
enter: true,
data: this.data,
el: this.targets[this.targetIndex].el,
sourceNode: this.sourceNode,
name: this.targets[this.targetIndex].name,
expand: this.targets[this.targetIndex].expand,
pageX: pageX,
pageY: pageY,
methods: methods
};
this.onDragOvers[this.targetIndex](params);
} else {
// 判断是否在目标内 是的话表示刚刚离开
if (this._inTarget) {
this._inTarget = false;
var _params2 = {
enter: false,
data: this.data,
el: this.targets[this._prevValidIndex].el,
sourceNode: this.sourceNode,
name: this.targets[this._prevValidIndex].name,
expand: this.targets[this._prevValidIndex].expand,
methods: methods
};
this.onDragLeaves[this._prevValidIndex](_params2);
}
}
},
/**
* 监听拖动结束
*/
onDragEnd: function onDragEnd() {
var _this2 = this;
// 触发放置事件
if (this.targetIndex >= 0) {
var params = {
enter: true,
data: this.data,
el: this.targets[this.targetIndex].el,
sourceNode: this.sourceNode,
name: this.targets[this.targetIndex].name,
expand: this.targets[this.targetIndex].expand,
methods: methods
};
this.targetIndex >= 0 && this.onDrops[this.targetIndex](params);
}
// 广播拖动结束事件
this.targetOnDragEnds.forEach(function (fn, index) {
if (!fn) return;
var params = {
enter: index === _this2.targetIndex,
data: _this2.data,
el: _this2.targets[index].el,
sourceNode: _this2.sourceNode,
name: _this2.targets[index].name,
expand: _this2.targets[index].expand,
methods: methods
};
fn(params);
});
},
/**
* 碰撞检测函数
*/
collision: function collision(pageX, pageY) {
var targetIndex = -1;
// 碰撞检测
for (var i = 0; i < this.targetPositions.length; i++) {
var position = this.targetPositions[i];
if (!position) continue;
var index = i;
if (pageX >= position.left && pageY >= position.top && pageX <= position.right && pageY <= position.bottom) {
targetIndex = index;
break;
}
}
return targetIndex;
},
/**
* 设置状态icon位置跟随鼠标
*/
setIconPosition: function setIconPosition(x, y) {
var style = this.stateIcon.style;
style.left = x + 8 + 'px';
style.top = y + 'px';
},
/**
* 显示状态icon
* url 可以是图片绝对路径 也可以是 add | error | delete
*/
showStateIcon: function showStateIcon(url) {
var _this3 = this;
setTimeout(function () {
if (_this3.isMobile) return console.warn('showStateIcon仅在pc端口可用 请参考相关说明' + _config.DOCUMENT_ADDR);
url = _iconImages2.default[url] || url || 'add';
var iconStyle = _this3.stateIcon.style;
iconStyle.display = 'block';
iconStyle.background = 'no-repeat url(' + url + ') center center / 100% auto';
}, 0);
},
/**
* 隐藏状态icon
*/
hideStateIcon: function hideStateIcon() {
try {
this.stateIcon.style.display = 'none';
} catch (e) {}
},
/**
* 移除被拖动的节点
* type 动画类型 不传则无动画直接消失 可选 fade | blost | back
* time 动画持续时长 非必填
*/
removeDragedNode: function removeDragedNode(type, time) {
var _this4 = this;
if (!type) return this.removeMark();
if (!_config.REMOVE_ANIMATION_TYPES[type]) return this.removeMark();
setTimeout(function () {
clearTimeout(_this4.removeMarkTid);
}, 0);
this[_config.REMOVE_ANIMATION_TYPES[type]](time);
}
}, _defineProperty(_dragStore, _config.REMOVE_ANIMATION_TYPES.fade, function () {
var time = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 150;
var style = this.draggedNode && this.draggedNode.style;
if (!style) return;
style.transition = 'all ' + time / 1000 + 's ease';
style.opacity = '0';
setTimeout(this.removeMark.bind(this), time);
}), _defineProperty(_dragStore, _config.REMOVE_ANIMATION_TYPES.blost, function () {
var time = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 150;
var style = this.draggedNode && this.draggedNode.style;
if (!style) return;
style.transition = 'all ' + time / 1000 + 's ease';
style.boxShadow = '0 0 50px 30px rgba(0,0,0,0.3)';
style.opacity = '0';
setTimeout(this.removeMark.bind(this), time);
}), _defineProperty(_dragStore, _config.REMOVE_ANIMATION_TYPES.back, function () {
var time = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 400;
var style = this.draggedNode && this.draggedNode.style;
if (!style) return;
style.transition = 'all ' + time / 1000 + 's cubic-bezier(0.2,0.4,0.25,1.1)';
style.transform = 'translate(0,0)';
setTimeout(this.removeMark.bind(this), time);
}), _defineProperty(_dragStore, 'removeMark', function removeMark() {
var _this5 = this;
clearTimeout(this.removeMarkTid);
this.removeMarkTid = setTimeout(function () {
try {
document.body.removeChild(_this5.markNode);
_this5.draggedNode = null;
} catch (e) {
// console.log('出错', e)
}
}, 10);
}), _defineProperty(_dragStore, 'destroyDrop', function destroyDrop(name) {
var _this6 = this;
var result = false;
this.targets.forEach(function (item, i) {
if (item.name === name) {
_this6.removeDrop(i);
result = true;
}
});
return result;
}), _defineProperty(_dragStore, 'removeDrop', function removeDrop(index) {
delete this.targets[index];
delete this.targetOnDragStarts[index];
delete this.targetOnDragEnds[index];
delete this.onDragEnters[index];
delete this.onDragLeaves[index];
delete this.onDragOvers[index];
delete this.onDrops[index];
delete this.targetPositions[index];
}), _defineProperty(_dragStore, 'getStateIconNode', function getStateIconNode() {
return this.stateIcon;
}), _dragStore);
exports.default = dragStore;
/**
* 供用户调用的静态方法
*/
var methods = exports.methods = {
showStateIcon: dragStore.showStateIcon.bind(dragStore),
hideStateIcon: dragStore.hideStateIcon.bind(dragStore),
getStateIconNode: dragStore.getStateIconNode.bind(dragStore),
removeDragedNode: dragStore.removeDragedNode.bind(dragStore),
destroyDrop: dragStore.destroyDrop.bind(dragStore)
};
/***/ }),
/* 2 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
exports.checkNode = checkNode;
function checkNode(el) {
var result = el;
if (!result) {
return console.error('找不到当前节点', el);
}
if (typeof el === 'string') {
var domName = el;
result = document.querySelector(domName);
if (!result) {
return console.error('找不到当前节点', el);
}
} else if ((typeof el === 'undefined' ? 'undefined' : _typeof(el)) === 'object') {
if (!el.nodeName) {
return console.error('找不到当前节点', el);
}
}
return result;
}
/***/ }),
/* 3 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _check = __webpack_require__(2);
var _store = __webpack_require__(1);
var _store2 = _interopRequireDefault(_store);
__webpack_require__(6);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
var Drag = function () {
function Drag(el, options) {
_classCallCheck(this, Drag);
this.initData(el, options) && this.init();
}
/**
* 检查和初始化传入参数
*/
_createClass(Drag, [{
key: 'initData',
value: function initData(el, options) {
this.el = (0, _check.checkNode)(el);
if (!this.el) return;
this.el.style.MozUserSelect = 'none'; // 兼容火狐
this.el.style.userSelect = 'none';
this.el.style.cursor = 'default';
options = this.checkOptions(options);
this.options = options;
this.data = options.data;
this.mouseDownPosition = { left: -1, top: -1 };
this.mouseDragging = false;
this.mark = null;
this.position = { left: 0, top: 0 };
return true;
}
// 初始化
}, {
key: 'init',
value: function init() {
this.addEventListenerPC();
this.addEventListenerMobile();
}
/**
* 事件监听
*/
}, {
key: 'addEventListenerPC',
value: function addEventListenerPC() {
var dom = this.el;
// 监听当前节点的鼠标点击事件
function onMouseDowm(e) {
var pageX = e.pageX,
pageY = e.pageY;
this.mouseDownPosition = { left: pageX, top: pageY };
dom.onmousemove = this.onElMousemove.bind(this);
dom.onmouseup = this.onElMouseUp.bind(this);
document.addEventListener('mouseup', this.onElMouseUp.bind(this));
}
dom.addEventListener('mousedown', onMouseDowm.bind(this));
}
}, {
key: 'addEventListenerMobile',
value: function addEventListenerMobile() {
// 兼容移动端
this.el.addEventListener('touchmove', this.onElTouchMove.bind(this));
this.el.addEventListener('touchend', this.onElTouchEnd.bind(this));
}
/**
* 监听拖动开始
*/
}, {
key: 'onElMousemove',
value: function onElMousemove(e) {
var pageX = e.pageX,
pageY = e.pageY;
var _mouseDownPosition = this.mouseDownPosition,
left = _mouseDownPosition.left,
top = _mouseDownPosition.top;
var EMIT_LENGTH = 3;
if (Math.abs(pageX - left) < EMIT_LENGTH && Math.abs(pageY - top) < EMIT_LENGTH) return;
if (this.mouseDragging) return;
this.mouseDragging = true;
_store2.default.onDragStart(this.data, this.el);
this.position = this.el.getBoundingClientRect();
// 创建蒙层
this.mark = document.createElement('div');
this.mark.className = 'x-drag-mark';
this.setMarkStyle();
this.mark.onmousemove = this.onMarkMouseMove.bind(this);
this.mark.onmouseup = this.onMarkMouseUp.bind(this);
this.mark.onmouseleave = this.onMarkMouseUp.bind(this);
_store2.default.markNode = this.mark;
document.body.appendChild(this.mark);
// 创建复制元素
_store2.default.draggedNode = this.el.cloneNode(true);
this.setCloneNodeStyle();
this.mark.appendChild(_store2.default.draggedNode);
// 创建状态icon
_store2.default.stateIcon = document.createElement('i');
_store2.default.stateIcon.className = 'x-state-icon';
this.setIconStyle();
this.mark.appendChild(_store2.default.stateIcon);
this.emit('onDragStart', {
el: this.el,
data: this.data,
methods: _store.methods
});
}
/**
* 监听拖动结束
*/
}, {
key: 'onElMouseUp',
value: function onElMouseUp() {
this.mouseDragging = false;
this.el.onmousemove = null;
this.el.onmouseup = null;
this.mark && (this.mark.onmousemove = null);
this.mark && (this.mark.onmouseup = null);
_store.methods.hideStateIcon();
_store.methods.removeDragedNode();
document.removeEventListener('mouseup', this.onElMouseUp.bind(this));
}
/**
* 监听蒙层鼠标移动
*/
}, {
key: 'onMarkMouseMove',
value: function onMarkMouseMove(e) {
if (!_store2.default.draggedNode) return;
var _ref = e.touches && e.touches[0] || e,
pageX = _ref.pageX,
pageY = _ref.pageY;
var translateX = pageX - this.mouseDownPosition.left;
var translateY = pageY - this.mouseDownPosition.top;
_store2.default.draggedNode.style.transform = 'translate(' + translateX + 'px,' + translateY + 'px)';
_store2.default.onDragOver(pageX, pageY);
}
/**
* 监听蒙层鼠标放开
*/
}, {
key: 'onMarkMouseUp',
value: function onMarkMouseUp() {
document.removeEventListener('mouseup', this.onElMouseUp.bind(this));
this.mouseDragging = false;
this.mark.onmousemove = null;
this.el.onmousemove = null;
this.mouseDownPosition = { left: -1, top: -1 };
this.emit('onDragEnd', {
el: this.el,
data: this.data,
target: _store2.default.targets[_store2.default.targetIndex],
methods: _store.methods
});
_store2.default.onDragEnd();
}
}, {
key: 'onElTouchMove',
value: function onElTouchMove(e) {
e.preventDefault();
_store2.default.isMobile = true;
if (this.mouseDownPosition.left === -1) {
var _e$touches$ = e.touches[0],
pageX = _e$touches$.pageX,
pageY = _e$touches$.pageY;
this.mouseDownPosition.left = pageX;
this.mouseDownPosition.top = pageY;
}
this.onElMousemove();
this.onMarkMouseMove(e);
}
}, {
key: 'onElTouchEnd',
value: function onElTouchEnd(e) {
this.onMarkMouseUp();
}
/**
* 检查并且初始化options
*/
}, {
key: 'checkOptions',
value: function checkOptions(options) {
options = options || {};
var baseOptions = {
data: '这里可以放需要丢给目标的内容',
el: this.el,
removeanimationtype: 1
};
for (var option in baseOptions) {
!options[option] && (options[option] = baseOptions[option]);
}
return options;
}
/**
* 设置蒙层样式
*/
}, {
key: 'setMarkStyle',
value: function setMarkStyle() {
var markStyle = {
position: 'absolute',
top: '0',
left: '0',
width: '100%',
height: '100%',
zIndex: '10'
};
for (var style in markStyle) {
this.mark.style[style] = markStyle[style];
}
}
/**
* 设置克隆节点样式
*/
}, {
key: 'setCloneNodeStyle',
value: function setCloneNodeStyle() {
var dom = _store2.default.draggedNode;
var style = dom.style;
var _position = this.position,
left = _position.left,
top = _position.top;
var _el$getBoundingClient = this.el.getBoundingClientRect(),
width = _el$getBoundingClient.width,
height = _el$getBoundingClient.height;
style.position = 'absolute';
style.left = left + 'px';
style.top = top + 'px';
style.width = width + 'px';
style.height = height + 'px';
style.textAlign = this.el.currentStyle.textAlign;
style.transform = 'translate(0,0)';
style.zIndex = 1000;
style.margin = 0;
}
/**
* 设置状态icon样式
*/
}, {
key: 'setIconStyle',
value: function setIconStyle() {
var style = _store2.default.stateIcon.style;
style.display = 'none';
style.position = 'absolute';
style.width = '20px';
style.height = '20px';
style.zIndex = '10001';
}
/**
* 发布事件
*/
}, {
key: 'emit',
value: function emit() {
var _options;
var args = Array.from(arguments);
var functionName = args.shift();
typeof this.options[functionName] === 'function' && (_options = this.options)[functionName].apply(_options, _toConsumableArray(args));
}
}]);
return Drag;
}();
module.exports = Drag;
/***/ }),
/* 4 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _check = __webpack_require__(2);
var _config = __webpack_require__(0);
var _store = __webpack_require__(1);
var _store2 = _interopRequireDefault(_store);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
var Drop = function () {
function Drop(el, options) {
_classCallCheck(this, Drop);
this.initData(el, options) && this.init();
}
/**
* 检查并初始化传入参数
*/
_createClass(Drop, [{
key: 'initData',
value: function initData(el, options) {
this.el = (0, _check.checkNode)(el);
if (!this.el) return;
this.options = this.checkOptions(options);
if (!this.options) return;
this.methods = _store.methods;
this.index = -1; // 当前索引
return true;
}
/**
* 检查并且初始化options
*/
}, {
key: 'checkOptions',
value: function checkOptions(options) {
if (!options) {
return console.error(this.el, '未检测到options 请参考相关说明' + _config.DOCUMENT_ADDR);
}
if (typeof options.onDrop !== 'function') {
return console.error(this.el, 'onDrop 必须是一个函数 请参考相关说明' + _config.DOCUMENT_ADDR);
}
var baseOptions = {
name: null
};
for (var option in baseOptions) {
!options[options] && (options[options] = baseOptions[option]);
}
return options;
}
/**
* 初始化
*/
}, {
key: 'init',
value: function init() {
this.setStore();
}
/**
* 托管状态
*/
}, {
key: 'setStore',
value: function setStore() {
var index = _store2.default.targets.push({
el: this.el,
name: this.options.name,
expand: this.options.expand
}) - 1;
this.index = index;
_store2.default.targetOnDragStarts[index] = this.onDragStart.bind(this);
_store2.default.targetOnDragEnds[index] = this.onDragEnd.bind(this);
_store2.default.onDragEnters[index] = this.onDragEnter.bind(this);
_store2.default.onDragOvers[index] = this.onDragOver.bind(this);
_store2.default.onDragLeaves[index] = this.onDragLeave.bind(this);
_store2.default.onDrops[index] = this.onDrop.bind(this);
}
/**
* 目标监听到拖动开始
*/
}, {
key: 'onDragStart',
value: function onDragStart(params) {
this.setStorePositions();
this.emit('onDragStart', params);
}
/**
* 目标监听到拖动结束
*/
}, {
key: 'onDragEnd',
value: function onDragEnd(params) {
this.emit('onDragEnd', params);
}
/**
* 目标监听到拖动进入当前范围
*/
}, {
key: 'onDragEnter',
value: function onDragEnter(params) {
this.emit('onDragEnter', params);
}
/**
* 目标监听在自己上方拖动
*/
}, {
key: 'onDragOver',
value: function onDragOver(params) {
this.emit('onDragOver', params);
}
/**
* 目标监听到离开当前范围
*/
}, {
key: 'onDragLeave',
value: function onDragLeave(params) {
this.emit('onDragLeave', params);
}
/**
* 目标监听到被拖动元素在自己范围内放下
*/
}, {
key: 'onDrop',
value: function onDrop(params) {
this.emit('onDrop', params);
}
/**
* 托管当前位置信息
*/
}, {
key: 'setStorePositions',
value: function setStorePositions() {
var _el$getBoundingClient = this.el.getBoundingClientRect(),
left = _el$getBoundingClient.left,
top = _el$getBoundingClient.top,
width = _el$getBoundingClient.width,
height = _el$getBoundingClient.height;
_store2.default.targetPositions[this.index] = {
top: top,
bottom: top + height,
left: left,
right: left + width
};
}
/**
* 发布事件
*/
}, {
key: 'emit',
value: function emit() {
var _options;
var args = Array.from(arguments);
var functionName = args.shift();
typeof this.options[functionName] === 'function' && (_options = this.options)[functionName].apply(_options, _toConsumableArray(args));
}
}]);
return Drop;
}();
module.exports = Drop;
/***/ }),
/* 5 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
var IMAGES = {
'add': '',
'error': '',
'delete': '',
'reject': ''
};
exports.default = IMAGES;
/***/ }),
/* 6 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
HTMLElement.prototype.__defineGetter__('currentStyle', function () {
return this.ownerDocument.defaultView.getComputedStyle(this, null);
});
/***/ }),
/* 7 */
/***/ (function(module, exports, __webpack_require__) {
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.Drop = exports.Drag = undefined;
var _drag = __webpack_require__(3);
var _drag2 = _interopRequireDefault(_drag);
var _drop = __webpack_require__(4);
var _drop2 = _interopRequireDefault(_drop);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
exports.Drag = _drag2.default;
exports.Drop = _drop2.default;
/***/ })
/******/ ]);
});
JavaScript实现拖放效果的更多相关文章
- javascript 拖放效果
最近一直在看javascript的书籍,有些东西在书上看着貌似理解了,但是在真正动手实践时,其实有些细节你根本不了解.所以看起来就算是一个简单的效果,写起来也未必简单,就算写起来简单,写的代码也未必规 ...
- 自学HTML5第三节(拖放效果)
今天来看看网页上的拖放效果,首先来看看什么是拖放———— 拖放 拖放是一种常见的特性,即抓取对象以后拖到另一个位置. 在 HTML5 中,拖放是标准的一部分,任何元素都能够拖放. 浏览器支持 Inte ...
- JavaScript 省市级联效果
JavaScript 省市级联效果 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" " ...
- Rainyday.js – 使用 JavaScript 实现雨滴效果
Rainyday.js 背后的想法是创建一个 JavaScript 库,利用 HTML5 Canvas 渲染一个雨滴落在玻璃表面的动画.Rainyday.js 有功能可扩展的 API,例如碰撞检测和易 ...
- JavaScript之放大镜效果2
在放大图片效果的同时,我们怎么原图和放大窗体增加间隔呢? 我们只需应用一个table就行了: 源码上: <!DOCTYPE html PUBLIC "-//W3C//DTD HTML ...
- JavaScript之放大镜效果
在网上也浏览过许多关于JavaScript放大镜效果的文章,有的代码解释得些隐晦难懂,看的我头有点晕晕的╮(╯﹏╰)╭,我的心情是这样的: 吐槽完了,我们动动小鼠标,当鼠标经过下面这张美女图片时就实现 ...
- JavaScript实现动画效果
说到JavaScript实现动画效果,不得不想到两个函数,setTimeout和setInterval. setTimeout的常用的使用方法为 setTimeout(callback, delay) ...
- javascript瀑布流效果
javascript瀑布流效果 其实javascript瀑布流 前几年都已经很流行了(特别是美丽说,蘑菇街),最近看到网上有人问这个瀑布流效果,所以自己有空的时候就研究了下,其实也是研究别人的代码,研 ...
- javascript跟随滚动效果插件代码(javascript Follow Plugin)
这篇文章介绍了javascript跟随滚动效果插件代码(javascript Follow Plugin),有需要的朋友可以参考一下Js 跟随滚动效果插件支持定义多个跟随ID,采用css fixed属 ...
随机推荐
- Anchor Boxes示例实战
Anchor Boxes示例实战 目标检测算法通常对输入图像中的大量区域进行采样,判断这些区域是否包含感兴趣的目标,并调整这些区域的边缘,以便更准确地预测目标的真实边界框.不同的模型可能使用不同的区域 ...
- 为已有数据的DataTable添加一个自增列
/// <summary> /// 为dt表增加一个自增的ID字段 /// </summary> /// <param name="dt">用户 ...
- JUC 并发编程--08,线程池,三大方法,七大参数,4种拒绝策略,代码演示
三大方法: //线程池核心线程数为n, 最大线程数为 n ExecutorService fixedThreadPool = Executors.newFixedThreadPool(n); 源码: ...
- 操作系统-gcc编译器驱动程序
gcc编译器驱动程序,读取x.c文件,翻译成可执行目标文件x 1.预处理阶段 预处理器(cpp)将x.c(源程序,文本文件)中的#等直接插入程序文本中,成为另一个c程序x.i(文本文件) 2.编译阶段 ...
- Django(64)频率认证源码分析与自定义频率认证
前言 有时候我们发送手机验证码,会发现1分钟只能发送1次,这是做了频率限制,限制的时间次数,都由开发者自己决定 频率认证源码分析 def check_throttles(self, request): ...
- 学习响应式编程 Reactor (5) - reactor 转换类操作符(2)
Reactor 操作符 上篇文章我们将 Flux 和 Mono 的操作符分了 11 类,我们来继续学习转换类操作符的第 2 篇. 转换类操作符 转换类的操作符数量最多,平常过程中也是使用最频繁的. F ...
- 树莓派FRP内网穿透及自启动
内网穿透的步骤和文件存档 实验室在远方部署了电脑主机来采集数据和图片,每次去调试会很麻烦,因而使用FRP内网穿透使得我们可以在实验室访问主机. 主要功能 实现远程可访问和开机自启FRP程序服务 安装和 ...
- STL----vector注意事项
开vector时要注意内存容易炸 最好的办法就是在开vector之后,对他进行一步操作 vector<int> a; a.resize(n); n就是你要开的数组的大小,此时数组里已经插入 ...
- gomod使用小结
gomod使用小结 使用方法 把工程拷贝到$GOPATH/src之外 在工程目录下执行:go mod init {module name}该命令会创建一个go.mod文件 然后在该目录下执行 go b ...
- react 工程目录简介
创建一个 todolist 项目,下图是其工程目录. node_modules文件夹 里面存放的是我们所建项目放所依赖的第三方的包 public文件夹 favicon.ico 图标文件,网页标题左上角 ...