代码之家  ›  专栏  ›  技术社区  ›  Psytronic

无法解析此引用的javascript对象方法

  •  0
  • Psytronic  · 技术社区  · 14 年前

    我已经创建了一个JS对象,有几个原型函数,从构造函数中调用它们是很好的,使用这个。[函数]

    但在后来的事件处理程序函数中,这指的是元素,而不是对象,我不确定如何解决这个问题:

    它通过addListener fine,鼠标按下事件触发器,使启用/禁用Ragging正常工作。但是在这个函数中,“this”是元素,而不是dragger对象,所以我不能调用access windowcenter、draggingbool或add/removelistener事件。

    从阅读来看,我可能需要实现prototype.bind?但不确定如何重新排列现有代码来执行此操作。还是要更改其中一个函数以返回函数?

    是的,我知道我可以在jquery中做这个,做很多次,只是想看看我是否能用pojs来实现它:)

    Dragger = function(element, draggingElement) {
        this.Element = element;
        this.DraggingElement = element;
        this.WindowCenter;
        this.DraggingBool = false;
    
        this.AddListener(this.DraggingElement, "mousedown", this.enableDragging); 
        this.AddListener(this.DraggingElement, "mouseup", this.disableDragging);
    }
    
    Dragger.prototype.AddListener = function (element, eventType, listener, bubble) {
        if (typeof (element) != "object" || typeof (listener) != "function")
            return;
        if (typeof (bubble) == "undefined")
            var bubble = false;
        eventType = eventType.toLowerCase();
        if (eventType.substr(0, 2) == "on")
            eventType = eventType.substr(2);
        if (window.addEventListener) { // Mozilla, Netscape, Firefox
            element.addEventListener(eventType, listener, bubble);
        } else if (window.attachEvent) { // IE
            element.attachEvent("on" + eventType, listener);
        }
    }
    
    Dragger.prototype.RemoveListener = function (element, eventType, listener) {
        eventType = eventType.toLowerCase();
        if (eventType.substr(0, 2) == "on")
            eventType = eventType.substr(2);
        if (window.addEventListener) { // Mozilla, Netscape, Firefox
            element.removeEventListener(eventType, listener, false);
        } else if (window.attachEvent) { // IE
            element.detachEvent("on" + eventType, listener);
        }
    }
    
    Dragger.prototype.enableDragging = function (e) {
        if (!e) var e = window.event;
        //dont move for right click
        if (e.which && e.which == 3)
            return;
        else if (e.button && e.button == 2)
            return;
    
        this.DraggingBool = true;
        this.WindowCenter = new Point(e.pageX, e.pageY);
        this.AddListener(document.body, "mouseover", this.mouseMoveListener);
    }
    
    Dragger.prototype.disableDragging = function () {
        this.DraggingBool = false;
        this.RemoveListener(document.body, "mouseover", this.mouseMoveListener);
    }
    
    Dragger.prototype.mouseMoveListener = function (e) {
        if (!this.DraggingBool)
            return;
        if (!e) var e = window.event;
        e.preventDefault();
        var newLoc = new Point(e.pageX, e.pageY);
        var xDif = windowCenter.X - newLoc.X;
        var yDif = windowCenter.Y - newLoc.Y;
        this.Element.style.left = (this.Element.offsetLeft + xDif) + "px";
        this.Element.style.top = (this.Element.offsetTop + yDif) + "px";
        this.WindowCenter = newLoc;
    }
    
    2 回复  |  直到 14 年前
        1
  •  2
  •   nas    14 年前

    addEventListener 执行范围被设置为节点,就像使用 node.onclick = foo . 当你使用 attachEvent 在IE中,执行范围变为全局执行范围,即窗口。

    function bind(fn, context) {
      return function() {
        return fn.apply(context, arguments);
      };
    }
    
    this.AddListener(this.DraggingElement, "mousedown",
        bind(this.enableDragging, this));
    

    您还可以向addListener函数添加一个上下文参数,并在其中进行绑定。

    Ecmascript 5 具有内置绑定方法。

        2
  •  0
  •   NM.    14 年前
      Function.prototype.bind = function () {
        if (arguments.length < 2 && arguments[0] === undefined) {
         return this;
       }
    
        var thisObj = this,
        args = Array.prototype.slice.call(arguments),
    
        obj = args.shift();
    
        return function () {
         return thisObj.apply(obj, args.concat(Array.prototype.slice.call(arguments)));
        };
    
      };
    
    
    
      Function.bind = function() {
        var args = Array.prototype.slice.call(arguments);
        return Function.prototype.bind.apply(args.shift(), args);
    
      }
    
    Dragger = function(element, draggingElement) {
        this.Element = element;
        this.DraggingElement = element;
        this.WindowCenter;
        this.DraggingBool = false;
    
        this.AddListener(this.DraggingElement, "mousedown", this.enableDragging,this); 
        this.AddListener(this.DraggingElement, "mouseup", this.disableDragging,this);
    }
    
    Dragger.prototype.AddListener = function (element, eventType, listener, bubble,dragger_obj) {
        if (typeof (element) != "object" || typeof (listener) != "function")
            return;
        if (typeof (bubble) == "undefined")
            var bubble = false;
        eventType = eventType.toLowerCase();
        if (eventType.substr(0, 2) == "on")
            eventType = eventType.substr(2);
        if (window.addEventListener) { // Mozilla, Netscape, Firefox
            element.addEventListener(eventType, listener.bind(dragger_obj), bubble);
        } else if (window.attachEvent) { // IE
            element.attachEvent("on" + eventType, listener.bind(dragger_obj));
        }
    }
    
    Dragger.prototype.RemoveListener = function (element, eventType, listener,dragger_obj) {
        eventType = eventType.toLowerCase();
        if (eventType.substr(0, 2) == "on")
            eventType = eventType.substr(2);
        if (window.addEventListener) { // Mozilla, Netscape, Firefox
            element.removeEventListener(eventType, listener.bind(dragger_obj), false);
        } else if (window.attachEvent) { // IE
            element.detachEvent("on" + eventType, listener.bind(dragger_obj));
        }
    }
    
    Dragger.prototype.enableDragging = function (e) {
        if (!e) var e = window.event;
        //dont move for right click
        if (e.which && e.which == 3)
            return;
        else if (e.button && e.button == 2)
            return;
    
        this.DraggingBool = true;
        this.WindowCenter = new Point(e.pageX, e.pageY);
        this.AddListener(document.body, "mouseover", this.mouseMoveListener,this);
    }
    
    Dragger.prototype.disableDragging = function () {
        this.DraggingBool = false;
        this.RemoveListener(document.body, "mouseover", this.mouseMoveListener,this);
    }
    
    Dragger.prototype.mouseMoveListener = function (e) {
        if (!this.DraggingBool)
            return;
        if (!e) var e = window.event;
        e.preventDefault();
        var newLoc = new Point(e.pageX, e.pageY);
        var xDif = windowCenter.X - newLoc.X;
        var yDif = windowCenter.Y - newLoc.Y;
        this.Element.style.left = (this.Element.offsetLeft + xDif) + "px";
        this.Element.style.top = (this.Element.offsetTop + yDif) + "px";
        this.WindowCenter = newLoc;
    }