拖拽系列二、利用JS面向对象OOP思想实现拖拽封装
接着上一篇拖拽系列一、JavaScript实现简单的拖拽效果这一篇博客将接着对上一节实现代码利用JS面向对象(OOP)思维对上一节代码进行封装;
使其模块化、避免全局函数污染、方便后期维护和调用;写到这里突然想起一句话“没有任何一个题目是彻底完成的。总还会有很多事情可做......”

我想这句话程序开发大概也适用吧,前端开发人员总是可以结合自己之前学到“拖拽”相关知识,不断扩展、完善、无穷无尽.......
利用匿名函数自执行实现封装
;(function(){
//do something......
})();
这样写的好处是避免全局变量、全局函数的污染;原因是函数存在作用域、作用域链、执行上下文;分号作用是避免代码压缩出错问题
面向对象(OOP)及代码大致结构简述
1.构造函数主要用于构造实例化的对象,每个对象都有自己特有(私有)的属性和方法(该属性和方法只供当前实例化对象调用);
2.对象原型通过调用构造函数实例化对象对属性与方法的实现共享
3.普通函数模块化
面向对象(OOP)思维分析拖拽:
重复说明——理解拖拽的核心就是掌握鼠标相关事件及鼠标事件相关联的操作流程
被拖拽的目标元素对象有
①鼠标按下时mousedown;
初始化鼠标的位置initMouseX、initMouseY;初始化目标元素位置initObjX、initObjY;鼠标按下标识isDraging
②鼠标移动时mousemove; 获取鼠标移动时的位置、计算目标元素的移动距离、设置目标元素的距离
③鼠标离开时mouseup; 停止移动,移除目标元素事件绑定;
代码大致结构如下
/*
* 利用OOP(面向对象) 实现拖拽代码的封装
*/
;(function(){
//事件处理程序
//elem DOM对象 eventName 事件名称 eventType 事件类型
function eventHandler(elem, eventName, eventType){};
//移除事件兼容处理
function removeEventHandler(elem, eventName, eventType){}
//获取style属性值
function getStyleValue(elem, property){}
//被拖拽构造函数
function Drag(selector){
//elem DOM对象
this.elem = typeof selector === 'object' ? selector : document.getElementById(selector);
//元素初始化位置
this.initObjX = 0;
this.initObjY = 0;
//鼠标初始化位置
this.initMouseX = 0;
this.initMouseY = 0;
this.isDraging = false;
//初始化--鼠标事件操作
this._init();
}
//Drag对象原型
Drag.prototype = {
constructor : Drag,
//初始化
//构造原型指回Drag 等价于==>>Drag.prototype._init = function(){}
_init : function(){
this.setDrag();
},
//获取目标元素pos位置
getObjPos : function(elem) {},
//设置被拖动元素的位置
setObjPos : function (elem, pos){},
//设置目标元素事件及操作流程
setDrag : function(){}
}
//将Drag挂到全局对象window上
window.Drag = Drag;
})();
实现拖拽功能函数的详述
上述较复杂思路集中在构造函数Drag创建被拖拽目标元素实例化对象,初始化设置目标元素鼠标事件及操作流程上,细节代码如下
//被拖拽构造函数
function Drag(selector){
//elem DOM对象
this.elem = typeof selector === 'object' ? selector : document.getElementById(selector);
//元素初始化位置
this.initObjX = 0;
this.initObjY = 0;
//鼠标初始化位置
this.initMouseX = 0;
this.initMouseY = 0;
this.isDraging = false;
//初始化--鼠标事件操作
this._init();
}
//Drag对象原型
Drag.prototype = {
constructor : Drag,
//初始化
//构造原型指回Drag 等价于==>>Drag.prototype._init = function(){}
_init : function(){
this.setDrag();
},
//设置目标元素事件及操作流程
setDrag : function(){
//目标元素对象
var self = this;
var time = null; //定时器
function mousedown(event){
event = window.event || event;
//鼠标按下时位置
this.initMouseX = event.clientX;
this.initMouseY = event.clientY;
//获取元素初始化位置pos
var pos = self.getObjPos(self.elem);
this.initObjX = pos.x;
this.initObjY = pos.y;
//mousemove
time = setTimeout(function(){ //缓解移动卡顿
eventHandler(self.elem, mousemove, "mousemove");
}, 25);
//mouseup
eventHandler(self.elem, mouseup, "mouseup");
//按下标识
self.isDraging = true;
}
function mousemove(event){
event = window.event || event;
if(self.isDraging){
//元素移动位置 == 当前鼠标移动位置 - 鼠标按下位置 + 目标元素初始化位置
var moveX = event.clientX - this.initMouseX + this.initObjX;
var moveY = event.clientY - this.initMouseY + this.initObjY;
//设置拖拽元素位置
self.setObjPos(self.elem, {
x : moveX,
y : moveY,
});
}
}
function mouseup(event){
event = window.event || event;
self.isDraging = false;
clearTimeout(time);
//移除事件
removeEventHandler(document, mousemove, 'mousemove');
removeEventHandler(document, mouseup, 'mouseup');
}
//mousedown
eventHandler(this.elem, mousedown, "mousedown");
}
}
至于设置/获取被拖拽的目标元素就很简单咯!
调用写法 new Drag(elem); elem为传入DOM对象即可
JS面向对象OOP实现拖拽完整代码
HTML代码
<style>
body {
margin: 0;
padding: 0;
position: relative;
}
.box {
width: 100px;
height: 100px;
background: deeppink;
position: absolute;
left: 25px;
top: 25px;
cursor: move;
}
</style>
<div class="box" id="box" style="position: absolute;left: 25px;top: 25px;"></div>
<script src="js/draging.js"></script>
<script type="text/javascript">
window.onload = function(){
new Drag(document.getElementById("box"));
}
</script>
draging.js
/*
* 利用JS面向对象OOP思想实现拖拽封装
*/
;(function(){
//事件处理程序
//elem DOM对象 eventName 事件名称 eventType 事件类型
function eventHandler(elem, eventName, eventType){
// elem.attachEvent 兼容IE9以下事件
elem.addEventListener ? elem.addEventListener(eventType, eventName, false) : elem.attachEvent('on'+eventType, eventName);
};
//移除事件兼容处理
function removeEventHandler(elem, eventName, eventType){
elem.removeEventListener ? elem.removeEventListener(eventType, eventName) : elem.detachEvent(eventType, eventName);
}
//获取style属性值
function getStyleValue(elem, property){
//getComputedStyle、currentStyle 返回CSS样式声明对象([object CSSStyleDeclaration]) 只读
//getComputedStyle 支持IE9+以上及正常浏览器
//currentStyle 兼容IE8及IE8以下获取目标元素style样式
return window.getComputedStyle(elem,null) ? window.getComputedStyle(elem,null)[property] : elem.currentStyle[property];
}
//被拖拽构造函数
function Drag(selector){
//elem DOM对象
this.elem = typeof selector === 'object' ? selector : document.getElementById(selector);
//元素初始化位置
this.initObjX = 0;
this.initObjY = 0;
//鼠标初始化位置
this.initMouseX = 0;
this.initMouseY = 0;
this.isDraging = false;
//初始化--鼠标事件操作
this._init();
}
//Drag对象原型
Drag.prototype = {
constructor : Drag,
//初始化
//构造原型指回Drag 等价于==>>Drag.prototype._init = function(){}
//初始化鼠标事件及鼠标操作流程
_init : function(){
this.setDrag();
},
//获取目标元素pos位置
getObjPos : function(elem) {
var pos = {x: 0, y: 0};
if(getStyleValue(elem, 'position') == 'static') {
this.elem.style.position = 'relative';
return pos;
} else {
var x = parseInt(getStyleValue(elem, 'left') ? getStyleValue(elem, 'left') : 0);
var y = parseInt(getStyleValue(elem, 'top') ? getStyleValue(elem, 'top') : 0);
return pos = {
x: x,
y: y
}
}
},
//设置被拖动元素的位置
setObjPos : function (elem, pos){
elem.style.position = 'absolute';
elem.style.left = pos.x+'px';
elem.style.top = pos.y+'px';
},
//设置目标元素事件及操作流程
setDrag : function(){
//目标元素对象
var self = this;
var time = null; //定时器
function mousedown(event){
event = window.event || event;
//鼠标按下时位置
this.initMouseX = event.clientX;
this.initMouseY = event.clientY;
//获取元素初始化位置pos
var pos = self.getObjPos(self.elem);
this.initObjX = pos.x;
this.initObjY = pos.y;
//mousemove
time = setTimeout(function(){ //缓解移动卡顿
eventHandler(self.elem, mousemove, "mousemove");
}, 25);
//mouseup
eventHandler(self.elem, mouseup, "mouseup");
//按下标识
self.isDraging = true;
}
function mousemove(event){
event = window.event || event;
if(self.isDraging){
//元素移动位置 == 当前鼠标移动位置 - 鼠标按下位置 + 目标元素初始化位置
var moveX = event.clientX - this.initMouseX + this.initObjX;
var moveY = event.clientY - this.initMouseY + this.initObjY;
//设置拖拽元素位置
self.setObjPos(self.elem, {
x : moveX,
y : moveY,
});
}
}
function mouseup(event){
event = window.event || event;
self.isDraging = false;
clearTimeout(time);
//移除事件
removeEventHandler(document, mousemove, 'mousemove');
removeEventHandler(document, mouseup, 'mouseup');
}
//mousedown
eventHandler(this.elem, mousedown, "mousedown");
}
}
//将Drag挂到全局对象window上
window.Drag = Drag;
})();
在线编辑代码请点击 http://jsrun.net/uukKp/edit
拖拽系列二、利用JS面向对象OOP思想实现拖拽封装的更多相关文章
- Java面向对象OOP思想概述
目录 OOP思想(Object Oriented Programming) 类和对象 接口 抽象类 OOP三大特性 封装 继承 多态 OOP复用的形式 OOP思想(Object Oriented Pr ...
- js面向对象oop编程
理解对象 对象这个词如雷贯耳,同样出名的一句话:XXX语言中一切皆为对象! 对象究竟是什么?什么叫面向对象编程? 对象(object),台湾译作物件,是面向对象(Object Oriented)中的术 ...
- Javascrpt 速成篇】 二:js面向对象
现实世界的对象由形态和行为组成,js中对应的是属性和函数. <!DOCTYPE html> <html> <head> <meta charset=" ...
- 面向对象oop思想
OOP核心思想:封装,继承,多态. 理解: 对象是由数据和容许的操作组成的封装体,与客观实体有直接对应关系,一个对象类定义了具有相似性质的一组对象.而每继承性是对具有层次关系的类的属性和操作进行共享的 ...
- JS面向对象编程(一):封装
js是一门基于面向对象编程的语言. 如果我们要把(属性)和(方法)封装成一个对象,甚至要从原型对象生成一个实例,我们应该怎么做呢? 一.生成对象的原始模式 假定把猫看 ...
- JavaScript面向对象OOP思想Class系统
JavaScript的Class模块,纯天然无依赖,只有2k大小,快速高效,让我们优雅的面向对象... | |目录 1源码:jClass.js 2源码:jClass.min.js 3构建一个类 4访问 ...
- js面向对象编程(一):封装(转载)
一. 生成对象的原始模式 假定我们把猫看成一个对象,它有"名字"和"颜色"两个属性. var Cat = { name : '', color : '' } 现 ...
- js面向对象编程思想
<!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...
- [js]面向对象编程
一.js面向对象基本概念 对象:内部封装.对外预留接口,一种通用的思想,面向对象分析: 1.特点 (1)抽象 (2)封装 (3)继承:多态继承.多重继承 2.对象组成 (1)属性: 任何对象都可以添加 ...
随机推荐
- firefox无法使用yslow的解决方案
首先,Yslow不支持firefox 36及以上版本. 解决方案:使用yslow的书签版本 使用方法:1.访问这里 http://yslow.org/mobile/ 2.把页面最后的那个 Deskto ...
- 腾讯云报告——MySQL成勒索新目标,数据服务基线安全问题迫在眉睫
推荐理由 大数据时代,人类产生的数据越来越多,但数据越多的情况下,也会带来数据的安全性问题,如MySQL数据库上的数据,越来越多的黑客盯上了它,今天推荐的这篇文章来自于腾讯云技术社区,主要是针对MyS ...
- css实现下拉菜单
实现一个效果不难,难的是使用最少的代码实现一个效果 <!DOCTYPE html> <html lang="en"> <head> <me ...
- python实现mysql的读写分离及负载均衡
Oracle数据库有其公司开发的配套rac来实现负载均衡,目前已知的最大节点数能到128个,但是其带来的维护成本无疑是很高的,并且rac的稳定性也并不是特别理想,尤其是节点很多的时候. 但是,相对my ...
- Zabbix3.0部署最佳实践
Zabbix3整个web界面做了一个全新的设计. 更多新特性请点击当前字幕查看 笔者QQ:572891887 Linux架构交流群:471443208 1.1Zabbix环境准备 [root@li ...
- Swift开发
1. 模糊效果 iconImageView.image = UIImage(named: "1.png") //效果类实例 let blurEffect = UIBlurEffec ...
- apache的配置参数
#ErrorDocument 500 "The server made a boo boo."#ErrorDocument 404 /missing.html 1.Document ...
- Centos 在 Xshell里 vim的配置
Centos里的VI只默认安装了vim-minimal-7.x.所以无论是输入vi或者vim查看文件,syntax功能都无法正常启用.因此需要用yum安装另外两个组件:vim-common-7.x和v ...
- JDBC基础学习(二)—PreparedStatement
一.PreparedStatement介绍 在SQL中包含特殊字符或SQL的关键字(如: ' or 1 or ')时Statement将出现不可预料的结果(出现异常或查询的结果不正确),可用P ...
- memcached的安装以及php两个扩展软件安装(memcache、memcached)
百度云安装包:http://pan.baidu.com/s/1pKZeDwn k3ap 1.安装memcached Memcached是基于libevent的事件处理,所以它的安装依赖libeven ...