在不丢失原型的情况下返回功能元素

时间:2013-11-03 16:37:52

标签: javascript prototypal-inheritance

有人知道如何创建一个返回元素而不会丢失原型的函数吗? 我正在尝试创建函数来创建一个新元素并将其作为元素返回。该函数将有一个操作该元素的方法。当我返回该元素时,原型将无法工作。如果我在该函数中不使用return,则原型正常工作,但该函数返回this对象。看看这段代码:

function ObjEl(tagName, id) {
    'use strict';
    this.node = document.createElement(tagName);
    if (typeof id === 'string' && id.match(/^[a-zA-Z\d\_\-]+$/)) {
        this.node.setAttribute('id', id);
    }

    // return this.node;
}

ObjEl.prototype.atr = function (key, val) {
    'use strict';
    this.node.setAttribute(key, val);
    return this;
};

如果我取消注释return this.node;,当我致电test = ObjEl('div', 'test');时,它会返回<div id="test"></div>,但此代码无效:

test = ObjEl('div', 'test').remove();

2 个答案:

答案 0 :(得分:0)

看起来很奇怪,但是这会实现你想要做的吗?

var ObjEl=function ObjEl(tagName, id) {
    'use strict';
    this.node = document.createElement(tagName);
    if (typeof id === 'string' && id.match(/^[a-zA-Z\d\_\-]+$/)) {
        this.node.setAttribute('id', id);
    }

    // return this.node;
}

ObjEl.prototype.atr = function (key, val) {
    'use strict';
    this.node.setAttribute(key, val);
    return this;
};
var oldObjEl=ObjEl;
ObjEl=function (){
  return new oldObjEl(arguments[0],arguments[1]);
}

答案 1 :(得分:0)

可能的选项

使用JavaScript实现这种行为的两种最佳方式(我知道)是创建一个包装对象来管理与内部相关元素的交易,即类似的东西到jQuery(见下面);或者直接使用您想要的额外内容扩展每个元素DOM Node。 e.g:

var extend = function( elm ){
  elm.attr = extend.attr;
};
extend.attr = function(){
  // your code here
};
var elm = extend(document.getElementById('div'));

可以也添加到某些元素的源构造函数的原型,即HTMLElement,但这不会跨浏览器工作,并可能导致其他代码出现问题。通常不赞成混淆基础构造函数。

我选择的方式

在过去,我使用以下粗略片段将多个不同的界面 - 有选择地 - 组合在一起 - 放在同一个对象中。然而,这不是经验丰富的代码,并且可以从正确的设置和拆除管理中获益 - 避免内存泄漏的风险 - 因为它基本上只是一组闭包。

var harmony = function(){
  if( !(this instanceof harmony) ) { return new harmony(); }
};
harmony.prototype.borrow = function( owner, descriptor ){
  var i, dlen = descriptor.length, item;
  for ( i=0; i<dlen; i++ ) {
    if ( (item = owner[descriptor[i]]) && item.apply ){
      this.borrowMethod( owner, descriptor[i] );
    }
    else {
      this.borrowAttribute( owner, descriptor[i] );
    }
  }
  return this;
};
harmony.prototype.add = function( owner, methods ){
  for ( var i in methods ) { this.addMethod( owner, i, methods[i] ); }
  return this;
};
harmony.prototype.addMethod = function( owner, name, method ){
  this[name] = function(){ return method.apply( owner, arguments ); };
  return this;
};
harmony.prototype.borrowMethod = function( owner, name ){
  this[name] = function(){ return owner[name].apply( owner, arguments ); };
  return this;
};
harmony.prototype.borrowAttribute = function( owner, attributeName ){
  var self = this; self[attributeName] = function(){
    if ( arguments.length ) { owner[attributeName] = arguments[0]; return self; }
    return owner[attributeName];
  };
  return self;
};

用法

以上内容可用于包装元素,从本机对象借用方法,以及使用自己的方法扩展。这是我的意思的粗略概念,这绝不是最终的代码。

var ElementHarmony = function(elm){
  var self = harmony()
    .borrow( elm, ['innerHTML', 'getElementsByTagName'] )
    .add( elm, {
      getElement: function(){
        return elm;
      },
      attr: function( name, value ){
        if ( arguments.length > 1 ) {
          this.setAttribute( name, value );
          return self;
        }
        else {
          return this.getAttribute( name );
        }
      }
    })
  ;
  return self;
}
/// create our example element and wrap with a harmony instance
var elm = ElementHarmony(document.createElement('span'))
  .innerHTML('<b>hello</b>')
  .attr('style', 'border: 1px solid red')
;
/// find the bold tag inside, and modify it's style
var bold = ElementHarmony(elm.getElementsByTagName('b')[0])
  .attr('style', 'background: orange')
;
/// use getElement() to get back to the original element,
/// when passing to native methods.
(document.body||document.documentElement).appendChild(elm.getElement());

使用和声对象,您可以从所有者对象借用方法和属性,但是属性将转换为方法,即innerHTML()。如果您知道您的代码只会在现代浏览器上执行,您可以更改此行为以使用getter和setter ......但是,这是另一个故事。

嘿呀呀呀

这是小提琴。

http://jsfiddle.net/K2Xux/