- 浏览: 314515 次
- 来自: ...
文章分类
最新评论
-
羊肉小笼包:
非常感谢,用第三种方式解决了service实现类里面装载不了d ...
autowire异常的三个情况 -
Rainyn:
碰到了另外一种情况。代码里有地方用到了dubbo的Servic ...
autowire异常的三个情况 -
tsinglongwu:
2.3都碰到过。但是今天发现一个新原因,就是mybatis忘了 ...
autowire异常的三个情况 -
cuqing:
@Servicepublic class RacePlanDa ...
autowire异常的三个情况 -
星丨澈:
帮了大忙了,用第三种方法解决了
autowire异常的三个情况
prototype 1.3.1 版本和之前的 1.2.0 版本有了不少改进,并增加了新的功能:
1. 增加了事件注册管理
2. 增加了空间定位的常用函数
3. 改善了 xmlhttp 的封装
4. 移除了 Effect.js,交给 Rico 或者 script.aculo.us 这些扩展库类实现。
5. bug 修复
1. 增加了事件注册管理
2. 增加了空间定位的常用函数
3. 改善了 xmlhttp 的封装
4. 移除了 Effect.js,交给 Rico 或者 script.aculo.us 这些扩展库类实现。
5. bug 修复
代码: |
/** * 定义一个全局对象, 属性 Version 在发布的时候会替换为当前版本号 */ var Prototype = { Version: '1.3.1', // 一个空方法,其后的代码常会用到,先前的版本该方法被定义于 Ajax 类中。 emptyFunction: function() {} } /** * 创建一种类型,注意其属性 create 是一个方法,返回一个构造函数。 * 一般使用如下 * var X = Class.create(); 返回一个类型,类似于 java 的一个Class实例。 * 要使用 X 类型,需继续用 new X()来获取一个实例,如同 java 的 Class.newInstance()方法。 * * 返回的构造函数会执行名为 initialize 的方法, initialize 是 Ruby 对象的构造器方法名字。 * 此时initialize方法还没有定义,其后的代码中创建新类型时会建立相应的同名方法。 * * 如果一定要从java上去理解。你可以理解为用Class.create()创建一个继承java.lang.Class类的类。当然java不允许这样做,因为Class类是final的 * */ var Class = { create: function() { return function() { this.initialize.apply(this, arguments); } } } /** * 创建一个对象,从变量名来思考,本意也许是定义一个抽象类,以后创建新对象都 extend 它。 * 但从其后代码的应用来看, Abstract 更多是为了保持命名空间清晰的考虑。 * 也就是说,我们可以给 Abstract 这个对象实例添加新的对象定义。 * * 从java去理解,就是动态给一个对象创建内部类。 */ var Abstract = new Object(); Object.extend = function(destination, source) { for (property in source) { destination[property] = source[property]; } return destination; } /** * 获取参数对象的所有属性和方法,有点象多重继承。但是这种继承是动态获得的。 * 如: * var a = new ObjectA(), b = new ObjectB(); * var c = a.extend(b); * 此时 c 对象同时拥有 a 和 b 对象的属性和方法。但是与多重继承不同的是,c instanceof ObjectB 将返回false。 * * 旧版本的该方法定义如下: * Object.prototype.extend = function(object) { * for (property in object) { * this[property] = object[property]; * } * return this; * } * * 新的形式新定义了一个静态方法 Object.extend,这样做的目的大概是为了使代码更为清晰 */ Object.prototype.extend = function(object) { return Object.extend.apply(this, [this, object]); } /** * 这个方法很有趣,它封装一个javascript函数对象,返回一个新函数对象,新函数对象的主体和原对象相同,但是bind()方法参数将被用作当前对象的对象。 * 也就是说新函数中的 this 引用被改变为参数提供的对象。 * 比如: * * * ................. * <script> * var aaa = document.getElementById("aaa"); * var bbb = document.getElementById("bbb"); * aaa.showValue = function() {alert(this.value);} * aaa.showValue2 = aaa.showValue.bind(bbb); * </script> * 那么,调用aaa.showValue 将返回"aaa", 但调用aaa.showValue2 将返回"bbb"。 * * apply 是ie5.5后才出现的新方法(Netscape好像很早就支持了)。 * 该方法更多的资料参考MSDN http://msdn.microsoft.com/library/en-us/script56/html/js56jsmthApply.asp * 阅读其后的代码就会发现,bind 被应用的很广泛,该方法和 Object.prototype.extend 一样是 Prototype 的核心。 * 还有一个 call 方法,应用起来和 apply 类似。可以一起研究下。 */ Function.prototype.bind = function(object) { var __method = this; return function() { __method.apply(object, arguments); } } /** * 和bind一样,不过这个方法一般用做html控件对象的事件处理。所以要传递event对象 * 注意这时候,用到了 Function.call。它与 Function.apply 的不同好像仅仅是对参数形式的定义。 * 如同 java 两个过载的方法。 */ Function.prototype.bindAsEventListener = function(object) { var __method = this; return function(event) { __method.call(object, event || window.event); } } /** * 将整数形式RGB颜色值转换为HEX形式 */ Number.prototype.toColorPart = function() { var digits = this.toString(16); if (this < 16) return '0' + digits; return digits; } /** * 典型 Ruby 风格的函数,将参数中的方法逐个调用,返回第一个成功执行的方法的返回值 */ var Try = { these: function() { var returnValue; for (var i = 0; i < arguments.length; i++) { var lambda = arguments[i]; try { returnValue = lambda(); break; } catch (e) {} } return returnValue; } } /*--------------------------------------------------------------------------*/ /** * 一个设计精巧的定时执行器 * 首先由 Class.create() 创建一个 PeriodicalExecuter 类型, * 然后用对象直接量的语法形式设置原型。 * * 需要特别说明的是 rgisterCallback 方法,它调用上面定义的函数原型方法bind, 并传递自己为参数。 * 之所以这样做,是因为 setTimeout 默认总以 window 对象为当前对象,也就是说,如果 registerCallback 方法定义如下的话: * registerCallback: function() { * setTimeout(this.onTimerEvent, this.frequency * 1000); * } * 那么,this.onTimeoutEvent 方法执行失败,因为它无法访问 this.currentlyExecuting 属性。 * 而使用了bind以后,该方法才能正确的找到this,也就是PeriodicalExecuter的当前实例。 */ var PeriodicalExecuter = Class.create(); PeriodicalExecuter.prototype = { initialize: function(callback, frequency) { this.callback = callback; this.frequency = frequency; this.currentlyExecuting = false; this.registerCallback(); }, registerCallback: function() { setInterval(this.onTimerEvent.bind(this), this.frequency * 1000); }, onTimerEvent: function() { if (!this.currentlyExecuting) { try { this.currentlyExecuting = true; this.callback(); } finally { this.currentlyExecuting = false; } } } } /*--------------------------------------------------------------------------*/ /** * 这个函数就 Ruby 了。我觉得它的作用主要有两个 * 1. 大概是 document.getElementById(id) 的最简化调用。 * 比如:$("aaa") 将返回 aaa 对象 * 2. 得到对象数组 * 比如: $("aaa","bbb") 返回一个包括id为"aaa"和"bbb"两个input控件对象的数组。 */ function $() { var elements = new Array(); for (var i = 0; i < arguments.length; i++) { var element = arguments[i]; if (typeof element == 'string') element = document.getElementById(element); if (arguments.length == 1) return element; elements.push(element); } return elements; } /** * 为兼容旧版本的浏览器增加 Array 的 push 方法。 */ if (!Array.prototype.push) { Array.prototype.push = function() { var startLength = this.length; for (var i = 0; i < arguments.length; i++) this[startLength + i] = arguments[i]; return this.length; } } /** * 为兼容旧版本的浏览器增加 Function 的 apply 方法。 */ if (!Function.prototype.apply) { // Based on code from http://www.youngpup.net/ Function.prototype.apply = function(object, parameters) { var parameterStrings = new Array(); if (!object) object = window; if (!parameters) parameters = new Array(); for (var i = 0; i < parameters.length; i++) parameterStrings[i] = 'parameters[' + i + ']'; object.__apply__ = this; var result = eval('object.__apply__(' + parameterStrings.join(', ') + ')'); object.__apply__ = null; return result; } } /** * 扩展 javascript 内置的 String 对象 */ String.prototype.extend({ /** * 去掉字符串中的标签 */ stripTags: function() { return this.replace(/<\/?[^>]+>/gi, ''); }, /** * 这个方法很常见,通常的实现都是用正则表达式替换特殊字符为html规范定义的命名实体或者十进制编码,比如: * string.replace(/&/g, "&").replace(//g, ">"); * 而这里的实现借用浏览器自身的内部替换,确实巧妙。 */ escapeHTML: function() { var div = document.createElement('div'); var text = document.createTextNode(this); div.appendChild(text); return div.innerHTML; }, /** * 同上 */ unescapeHTML: function() { var div = document.createElement('div'); div.innerHTML = this.stripTags(); return div.childNodes[0].nodeValue; } }); /** * 定义 Ajax 对象, 静态方法 getTransport 方法返回一个 XMLHttp 对象 */ var Ajax = { getTransport: function() { return Try.these( function() {return new ActiveXObject('Msxml2.XMLHTTP')}, function() {return new ActiveXObject('Microsoft.XMLHTTP')}, function() {return new XMLHttpRequest()} ) || false; } } /** * 我以为此时的Ajax对象起到命名空间的作用。 * Ajax.Base 声明为一个基础对象类型 * 注意 Ajax.Base 并没有使用 Class.create() 的方式来创建,我想是因为作者并不希望 Ajax.Base 被库使用者实例化。 * 作者在其他对象类型的声明中,将会继承于它。 * 就好像 java 中的私有抽象类 */ Ajax.Base = function() {}; Ajax.Base.prototype = { /** * extend (见上) 的用法真是让人耳目一新 * options 首先设置默认属性,然后再 extend 参数对象,那么参数对象中也有同名的属性,那么就覆盖默认属性值。 * 想想如果我写这样的实现,应该类似如下: setOptions: function(options) { this.options.methed = options.methed? options.methed : 'post'; .......... } 我想很多时候,java 限制了 js 的创意。 */ setOptions: function(options) { this.options = { method: 'post', asynchronous: true, parameters: '' }.extend(options || {}); }, /** * 如果 xmlhttp 调用返回正确的HTTP状态值,函数返回ture, 反之false。 * xmlhttp 的 readyState 属性不足以准确判断 xmlhttp 远程调用成功,该方法是readyState判断的一个前提条件 */ responseIsSuccess: function() { return this.transport.status == undefined || this.transport.status == 0 || (this.transport.status >= 200 && this.transport.status < 300); }, /** * 如果 xmlhttp 调用返回错误的HTTP状态值,函数返回ture, 反之false。 */ responseIsFailure: function() { return !this.responseIsSuccess(); } } /** * Ajax.Request 封装 XmlHttp */ Ajax.Request = Class.create(); /** * 定义四种事件(状态), 参考http://msdn.microsoft.com/workshop/author/dhtml/reference/properties/readystate_1.asp */ Ajax.Request.Events = ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete']; /** * 相比先前的版本,对于 xmlhttp 的调用和返回值处理分离得更为清晰 */ Ajax.Request.prototype = (new Ajax.Base()).extend({ initialize: function(url, options) { this.transport = Ajax.getTransport(); this.setOptions(options); this.request(url); }, /** * 新增加 request 方法封装 xmlhttp 的调用过程。 */ request: function(url) { var parameters = this.options.parameters || ''; if (parameters.length > 0) parameters += '&_='; try { if (this.options.method == 'get') url += '?' + parameters; this.transport.open(this.options.method, url, this.options.asynchronous); if (this.options.asynchronous) { this.transport.onreadystatechange = this.onStateChange.bind(this); setTimeout((function() {this.respondToReadyState(1)}).bind(this), 10); } this.setRequestHeaders(); var body = this.options.postBody ? this.options.postBody : parameters; this.transport.send(this.options.method == 'post' ? body : null); } catch (e) { } }, /** * 新增加的 setRequestHeaders 方法允许添加自定义的http header */ setRequestHeaders: function() { var requestHeaders = ['X-Requested-With', 'XMLHttpRequest', 'X-Prototype-Version', Prototype.Version]; if (this.options.method == 'post') { requestHeaders.push('Content-type', 'application/x-www-form-urlencoded'); /* Force "Connection: close" for Mozilla browsers to work around * a bug where XMLHttpReqeuest sends an incorrect Content-length * header. See Mozilla Bugzilla #246651. */ if (this.transport.overrideMimeType) requestHeaders.push('Connection', 'close'); } /** * 其后的 apply 方法的调用有些奇技淫巧的意味 * 从上下文中我们可以分析出 this.options.requestHeaders 是调用者自定义的http header数组。 * requestHeaders 也是一个数组,将一个数组中的元素逐个添加到另一个元素中,直接调用 * requestHeaders.push(this.options.requestHeaders) * 是不行的,因为该调用导致 this.options.requestHeaders 整个数组作为一个元素添加到 requestHeaders中。 * javascript的Array对象还提供一个concat 的方法表面上满足要求,但是concat实际上是创建一个新数组,将两个数组的元素添加到新数组中。 * 所以,下面的代码也可以替换为 * requestHeaders = requestHeaders.concat(this.options.requestHeaders); * 很显然,作者不喜欢这样的代码方式 * 而 apply 方法的语法 apply([thisObj[,argArray]]) 本身就要求第二个参数是一个数组或者arguments对象。 * 所以巧妙的实现了 concat 函数的作用。 * 令人拍案叫绝啊! */ if (this.options.requestHeaders) requestHeaders.push.apply(requestHeaders, this.options.requestHeaders); for (var i = 0; i < requestHeaders.length; i += 2) this.transport.setRequestHeader(requestHeaders[i], requestHeaders[i+1]); }, onStateChange: function() { var readyState = this.transport.readyState; /** * 如果不是 Loading 状态,就调用回调函数 */ if (readyState != 1) this.respondToReadyState(this.transport.readyState); }, /** * 回调函数定义在 this.options 属性中,比如: var option = { onLoaded : function(req) {...}; ...... } new Ajax.Request(url, option); */ respondToReadyState: function(readyState) { var event = Ajax.Request.Events[readyState]; /** * 新增的回调函数处理,调用者还可以在options中定义 on200, onSuccess 这样的回调函数 * 在 readyState 为完成状态的时候调用 */ if (event == 'Complete') (this.options['on' + this.transport.status] || this.options['on' + (this.responseIsSuccess() ? 'Success' : 'Failure')] || Prototype.emptyFunction)(this.transport); (this.options['on' + event] || Prototype.emptyFunction)(this.transport); /* Avoid memory leak in MSIE: clean up the oncomplete event handler */ if (event == 'Complete') this.transport.onreadystatechange = Prototype.emptyFunction; } }); /** * Ajax.Updater 用于绑定一个html元素与 XmlHttp调用的返回值。类似与 buffalo 的 bind。 * 如果 options 中有 insertion(见后) 对象的话, insertion 能提供更多的插入控制。 */ Ajax.Updater = Class.create(); Ajax.Updater.ScriptFragment = '(?:<script.*?>)((\n|.)*?)(?:<\/script>)'; Ajax.Updater.prototype.extend(Ajax.Request.prototype).extend({ initialize: function(container, url, options) { /** * containers 就是被绑定的 html 对象,xmlhttp的返回值被赋给该对象的 innerHTML 属性。 * 相比新版本,containers 根据container参数定义 success 和 failure 引用,如果它们被定义的话,根据xmlhttp调用是否成功来选择 * 更新对象,假想调用可能如下: * var c = {success: $("successDiv"), failure: $("failureDiv")}; * new Ajax.Updater(c, url, options); * 那么调用成功则 successDiv 显示成功信息或者数据,反之 failureDiv 显示错误信息 */ this.containers = { success: container.success ? $(container.success) : $(container), failure: container.failure ? $(container.failure) : (container.success ? null : $(container)) } this.transport = Ajax.getTransport(); this.setOptions(options); var onComplete = this.options.onComplete || Prototype.emptyFunction; this.options.onComplete = (function() { this.updateContent(); onComplete(this.transport); }).bind(this); this.request(url); }, updateContent: function() { var receiver = this.responseIsSuccess() ? this.containers.success : this.containers.failure; var match = new RegExp(Ajax.Updater.ScriptFragment, 'img'); var response = this.transport.responseText.replace(match, ''); var scripts = this.transport.responseText.match(match); if (receiver) { if (this.options.insertion) { new this.options.insertion(receiver, response); } else { receiver.innerHTML = response; } } if (this.responseIsSuccess()) { if (this.onComplete) setTimeout((function() {this.onComplete( this.transport)}).bind(this), 10); } /** * 如果调用者在传入的options参数中定义 evalScripts=true,同时xmlhttp返回值的html中包含<script>标签的话,执行该脚本 */ if (this.options.evalScripts && scripts) { /** * 注意前二十行左右还有一个 match 的声明 * var match = new RegExp(Ajax.Updater.ScriptFragment, 'img'); * 和此处的区别就是,正则表达式匹配标记多一个 "g"。 * 多个g, 所以 scripts 是一个数组,数组中每个元素是一段 <script>...</script> 文本。 * 没有g, scripts[i].match(match)[1] 匹配的就是 <script type="text/javascript"></script> 标记中的 script 代码。 * 关于正则表达式,请参考javascript的相关资料。 */ match = new RegExp(Ajax.Updater.ScriptFragment, 'im'); setTimeout((function() { for (var i = 0; i < scripts.length; i++) eval(scripts[i].match(match)[1]); }).bind(this), 10); } } }); /** * 定期更新器 */ Ajax.PeriodicalUpdater = Class.create(); Ajax.PeriodicalUpdater.prototype = (new Ajax.Base()).extend({ initialize: function(container, url, options) { this.setOptions(options); this.onComplete = this.options.onComplete; this.frequency = (this.options.frequency || 2); // decay 可能是一个时间调整因素 this.decay = 1; this.updater = {}; this.container = container; this.url = url; this.start(); }, start: function() { this.options.onComplete = this.updateComplete.bind(this); this.onTimerEvent(); }, stop: function() { this.updater.onComplete = undefined; clearTimeout(this.timer); (this.onComplete || Ajax.emptyFunction).apply(this, arguments); }, updateComplete: function(request) { if (this.options.decay) { this.decay = (request.responseText == this.lastText ? this.decay * this.options.decay : 1); this.lastText = request.responseText; } this.timer = setTimeout(this.onTimerEvent.bind(this), this.decay * this.frequency * 1000); }, onTimerEvent: function() { this.updater = new Ajax.Updater(this.container, this.url, this.options); } }); /** * 根据 class attribute 的名字得到对象数组,支持 multiple class * */ document.getElementsByClassName = function(className) { var children = document.getElementsByTagName('*') || document.all; var elements = new Array(); for (var i = 0; i < children.length; i++) { var child = children[i]; var classNames = child.className.split(' '); for (var j = 0; j < classNames.length; j++) { if (classNames[j] == className) { elements.push(child); break; } } } return elements; } /*--------------------------------------------------------------------------*/ /** * Element 就象一个 java 的工具类,主要用来 隐藏/显示/销除 对象,以及获取对象的简单属性。 * */ if (!window.Element) { var Element = new Object(); } Object.extend(Element, { /** * 切换 显示/隐藏 */ toggle: function() { for (var i = 0; i < arguments.length; i++) { var element = $(arguments[i]); element.style.display = (element.style.display == 'none' ? '' : 'none'); } }, hide: function() { for (var i = 0; i < arguments.length; i++) { var element = $(arguments[i]); element.style.display = 'none'; } }, show: function() { for (var i = 0; i < arguments.length; i++) { var element = $(arguments[i]); element.style.display = ''; } }, /** * 从父节点中移除 */ remove: function(element) { element = $(element); element.parentNode.removeChild(element); }, getHeight: function(element) { element = $(element); return element.offsetHeight; }, /** * 是否拥有 class 属性值 */ hasClassName: function(element, className) { element = $(element); if (!element) return; var a = element.className.split(' '); for (var i = 0; i < a.length; i++) { if (a[i] == className) return true; } return false; }, /** * 为对象添加 class 属性值 */ addClassName: function(element, className) { element = $(element); Element.removeClassName(element, className); element.className += ' ' + className; }, /** * 为对象移除 class 属性值 */ removeClassName: function(element, className) { element = $(element); if (!element) return; var newClassName = ''; var a = element.className.split(' '); for (var i = 0; i < a.length; i++) { if (a[i] != className) { if (i > 0) newClassName += ' '; newClassName += a[i]; } } element.className = newClassName; }, // removes whitespace-only text node children cleanWhitespace: function(element) { var element = $(element); for (var i = 0; i < element.childNodes.length; i++) { var node = element.childNodes[i]; if (node.nodeType == 3 && !/\S/.test(node.nodeValue)) Element.remove(node); } } }); /** * 为 Element.toggle 做了一个符号连接,大概是兼容性的考虑 */ var Toggle = new Object(); Toggle.display = Element.toggle; /*--------------------------------------------------------------------------*/ /** * 动态插入内容的实现,MS的Jscript实现中对象有一个 insertAdjacentHTML 方法(http: //msdn.microsoft.com/workshop/author/dhtml/reference/methods/insertadjacenthtml.asp) * 这里算是一个对象形式的封装。 */ Abstract.Insertion = function(adjacency) { this.adjacency = adjacency; } Abstract.Insertion.prototype = { initialize: function(element, content) { this.element = $(element); this.content = content; if (this.adjacency && this.element.insertAdjacentHTML) { this.element.insertAdjacentHTML(this.adjacency, this.content); } else { /** * gecko 不支持 insertAdjacentHTML 方法,但可以用如下代码代替 */ this.range = this.element.ownerDocument.createRange(); /** * 如果定义了 initializeRange 方法,则实行,这里相当与定义了一个抽象的 initializeRange 方法 */ if (this.initializeRange) this.initializeRange(); this.fragment = this.range.createContextualFragment(this.content); /** * insertContent 也是一个抽象方法,子类必须实现 */ this.insertContent(); } } } /** * prototype 加深了我的体会,就是写js 如何去遵循 Don’t Repeat Yourself (DRY) 原则 * 上文中 Abstract.Insertion 算是一个抽象类,定义了名为 initializeRange 的一个抽象方法 * var Insertion = new Object() 建立一个命名空间 * Insertion.Before|Top|Bottom|After 就象是四个java中的四个静态内部类,而它们分别继承于Abstract.Insertion,并实现了initializeRange方法。 */ var Insertion = new Object(); /** * 将内容插入到指定节点的前面, 与指定节点同级 */ Insertion.Before = Class.create(); Insertion.Before.prototype = (new Abstract.Insertion('beforeBegin')).extend({ initializeRange: function() { this.range.setStartBefore(this.element); }, insertContent: function() { this.element.parentNode.insertBefore(this.fragment, this.element); } }); /** * 将内容插入到指定节点的第一个子节点前,于是内容变为该节点的第一个子节点 */ Insertion.Top = Class.create(); Insertion.Top.prototype = (new Abstract.Insertion('afterBegin')).extend({ initializeRange: function() { this.range.selectNodeContents(this.element); this.range.collapse(true); }, insertContent: function() { this.element.insertBefore(this.fragment, this.element.firstChild); } }); /** * 将内容插入到指定节点的最后,于是内容变为该节点的最后一个子节点 */ Insertion.Bottom = Class.create(); Insertion.Bottom.prototype = (new Abstract.Insertion('beforeEnd')).extend({ initializeRange: function() { this.range.selectNodeContents(this.element); this.range.collapse(this.element); }, insertContent: function() { this.element.appendChild(this.fragment); } }); /** * 将内容插入到指定节点的后面, 与指定节点同级 */ Insertion.After = Class.create(); Insertion.After.prototype = (new Abstract.Insertion('afterEnd')).extend({ initializeRange: function() { this.range.setStartAfter(this.element); }, insertContent: function() { this.element.parentNode.insertBefore(this.fragment, this.element.nextSibling); } }); /** * 针对 页面元素对象(一般都是表单控件)的工具类,提供一些简单静态方法 * 这些方法显然常用在表单处理中 * 注意 Field 这种声明方式类似于 java 声明一个静态的 singleton 工具类 * 等同于 : * var Field = new Object(); * Field.extend({...}); * * 后文中的 Form, Event, Position 对象声明方式如出一辙 */ var Field = { /** * 清除参数引用的对象的值 */ clear: function() { for (var i = 0; i < arguments.length; i++) $(arguments[i]).value = ''; }, /** * 使参数引用对象获取焦点 */ focus: function(element) { $(element).focus(); }, /** * 判断参数引用对象是否有非空值,如为空值,返回false, 反之true */ present: function() { for (var i = 0; i < arguments.length; i++) if ($(arguments[i]).value == '') return false; return true; }, /** * 使选中参数引用对象 */ select: function(element) { $(element).select(); }, /** * 使参数引用对象处于可编辑状态 */ activate: function(element) { $(element).focus(); $(element).select(); } } /*--------------------------------------------------------------------------*/ /** * 表单工具类 */ var Form = { /** * 将表单元素序列化后的值(其实就是 name=value 形式的名值配对)组合成 QueryString 的形式 */ serialize: function(form) { var elements = Form.getElements($(form)); var queryComponents = new Array(); for (var i = 0; i < elements.length; i++) { var queryComponent = Form.Element.serialize(elements[i]); if (queryComponent) queryComponents.push(queryComponent); } return queryComponents.join('&'); }, /** * 得到表单的所有元素对象 */ getElements: function(form) { var form = $(form); var elements = new Array(); for (tagName in Form.Element.Serializers) { var tagElements = form.getElementsByTagName(tagName); for (var j = 0; j < tagElements.length; j++) elements.push(tagElements[j]); } return elements; }, /** * 根据 type 和 name 过滤得到表单中符合的 对象 */ getInputs: function(form, typeName, name) { var form = $(form); var inputs = form.getElementsByTagName('input'); if (!typeName && !name) return inputs; var matchingInputs = new Array(); for (var i = 0; i < inputs.length; i++) { var input = inputs[i]; if ((typeName && input.type != typeName) || (name && input.name != name)) continue; matchingInputs.push(input); } return matchingInputs; }, /** * 将指定表单的元素置于不可用状态 */ disable: function(form) { var elements = Form.getElements(form); for (var i = 0; i < elements.length; i++) { var element = elements[i]; element.blur(); element.disabled = 'true'; } }, /** * 将指定表单的元素置于可用状态 */ enable: function(form) { var elements = Form.getElements(form); for (var i = 0; i < elements.length; i++) { var element = elements[i]; element.disabled = ''; } }, /** * 使表单的第一个非 hidden 类型而且处于可用状态的元素获得焦点 */ focusFirstElement: function(form) { var form = $(form); var elements = Form.getElements(form); for (var i = 0; i < elements.length; i++) { var element = elements[i]; if (element.type != 'hidden' && !element.disabled) { Field.activate(element); break; } } }, /* * 重置表单 */ reset: function(form) { $(form).reset(); } } /** * 表单元素工具类 */ Form.Element = { /** * 返回表单元素的值先序列化, 其实就是 name=value 形式的名值配对 */ serialize: function(element) { var element = $(element); var method = element.tagName.toLowerCase(); var parameter = Form.Element.Serializers[method](element); if (parameter) return encodeURIComponent(parameter[0]) + '=' + encodeURIComponent(parameter[1]); }, /** * 返回表单元素的值 */ getValue: function(element) { var element = $(element); var method = element.tagName.toLowerCase(); var parameter = Form.Element.Serializers[method](element); if (parameter) return parameter[1]; } } /** * prototype 的所谓序列化其实就是将表单的名字和值组合成一个数组 */ Form.Element.Serializers = { input: function(element) { switch (element.type.toLowerCase()) { case 'submit': case 'hidden': case 'password': case 'text': return Form.Element.Serializers.textarea(element); case 'checkbox': case 'radio': return Form.Element.Serializers.inputSelector(element); } return false; }, /** * 单/多选框 由此方法处理序列化 */ inputSelector: function(element) { if (element.checked) return [element.name, element.value]; }, /** * textarea 由此方法处理序列化 */ textarea: function(element) { return [element.name, element.value]; }, /** * select 下拉列表由此方法处理序列化 */ select: function(element) { var value = ''; if (element.type == 'select-one') { var index = element.selectedIndex; if (index >= 0) value = element.options[index].value || element.options[index].text; } else { /** * 支持 select-mulitple 的下拉列表,返回的数组的第二个元素,是一个值数组 */ value = new Array(); for (var i = 0; i < element.length; i++) { var opt = element.options[i]; if (opt.selected) value.push(opt.value || opt.text); } } return [element.name, value]; } } /*--------------------------------------------------------------------------*/ /** * Form.Element.getValue 会经常用到,所以做了一个快捷引用 * 取得某个表单控件的值,可以简化调用为 $F("username"),真是方便啊 */ var $F = Form.Element.getValue; /*--------------------------------------------------------------------------*/ /** * Abstract.TimedObserver 也没有用 Class.create() 来创建,和Ajax.Base 意图应该一样 * Abstract.TimedObserver 顾名思义,是套用Observer设计模式来跟踪指定表单元素, * 当表单元素的值发生变化的时候,就执行回调函数 * * 我想 Observer 与注册onchange事件相似,不同点在于 onchange 事件是在元素失去焦点的时候才激发。 * 同样的与 onpropertychange 事件也相似,不过它只关注表单元素的值的变化,而且提供timeout的控制。 * * 除此之外,Observer 的好处大概就在与更面向对象,另外可以动态的更换回调函数,这就比注册事件要灵活一些。 * Observer 应该可以胜任动态数据校验,或者多个关联下拉选项列表的连动等等 * */ Abstract.TimedObserver = function() {} /** * 这个设计和 PeriodicalExecuter 一样,bind 方法是实现的核心 */ Abstract.TimedObserver.prototype = { initialize: function(element, frequency, callback) { this.frequency = frequency; this.element = $(element); this.callback = callback; this.lastValue = this.getValue(); this.registerCallback(); }, registerCallback: function() { &nb |
发表评论
-
jquery-easyui-datagrid中的问题改进,关于一次性查询和分页查询的问题
2010-02-25 15:58 9446datagrid可以实现服务器上分页,但是不管怎样分页都要到数 ... -
jquery
2009-10-28 14:56 982在用jquery中的div.load(url)中,如果是ses ... -
各种效果的jquery ui(接口)介绍
2009-09-28 16:14 1025基本的鼠标 互动: 拖拽(drag and droppin ... -
javascript表单之间的数据传递
2007-05-09 13:42 2149xml 代码 javascript表单之间的 ... -
prototype.js用法
2007-01-29 18:11 1257http://blog.csdn.net/jxufewbt/a ... -
lightbox
2007-01-16 15:05 1382“Lightbox”是一个别致且易用的图片显示效果,它可以使 ... -
prototype.js常用函数
2007-01-16 14:39 1444prototype.js常用函数: ... -
常规函数
2007-01-16 11:35 13171.常规函数 javascript常规函数包括以下9 ...
相关推荐
《prototype_1.7.3.js:JavaScript框架的里程碑》 在JavaScript的世界里,Prototype库是一个不可或缺的重要组成部分,尤其在Web开发领域,它为开发者提供了强大的功能和便利性。Prototype_1.7.3.js是这个库的一个...
通过学习和使用prototype.js,开发者可以编写出结构更清晰、可维护性更强的JavaScript代码,同时也能提高开发效率,减少跨浏览器兼容性的烦恼。然而,随着ES6及后续版本的推出,许多prototype.js的功能已被原生...
Prototype.js的核心理念是通过扩展JavaScript的内置类型和添加实用工具方法,使开发者能更高效地编写代码。 一、Prototype.js的核心特性 1. **DOM操作**:Prototype.js为DOM元素提供了丰富的操作接口,如`Element....
9. **性能优化**:Prototype.js 在设计时就考虑了性能,通过缓存DOM查找结果、延迟执行等手段,提高了代码执行效率。 10. **兼容性**:Prototype.js 兼容多种浏览器,包括IE6+、Firefox、Chrome、Safari和Opera等,...
- **对象扩展**:`prototype.js`提供了对JavaScript原生对象的扩展,如Array、String、Function等,增加了许多实用的方法,使得代码编写更为简洁。 - **DOM操作**:库中包含了一系列方便的DOM操作方法,如`Element....
《Prototype.js 1.4-1.6:JavaScript 动态原型框架的探索与实践》 Prototype.js 是一个广泛使用的JavaScript库,它扩展了JavaScript语言的功能,为开发人员提供了更强大的面向对象编程支持。该库的核心特性是其对...
Prototype.js 是一个开源的JavaScript库,它扩展了JavaScript语言,为开发者提供了许多便利的功能,特别是在对象操作、DOM操作和事件处理方面。1.6版本是该库的一个重要里程碑,引入了诸多改进和新特性。 ### 1. ...
在实际开发中,Prototype.js可以显著提高开发效率,减少代码量,并且由于其良好的兼容性和丰富的功能,成为了许多开发者喜爱的JavaScript插件。下载并使用"prototype.js"文件,即可开始体验这些强大的功能。不过要...
`prototypeAPI` 文件可能是 Prototype.js 1.6 的 API 文档,包含了详细的函数说明、参数和示例。通过查阅这份文档,开发者可以更全面地了解每个方法的用法,以便在实际项目中有效地应用。 综上所述,Prototype.js ...
它在Web开发中扮演着重要的角色,尤其在AJAX(异步JavaScript和XML)应用中,Prototype.js 提供了丰富的功能和工具,使得开发者能更高效地编写JavaScript代码。 ### 一、Prototype.js核心特性 1. **DOM操作**:...
通过学习《prototype.js 1.4版开发者手册》,开发者可以掌握这些高级技巧,提高开发效率,编写出更加优雅和高效的JavaScript代码。同时,分享这份手册也是促进社区交流和共同进步的好方式,因为学习从来不是孤立的...
在JavaScript的世界里,Prototype.js是一个著名的库,它扩展了JavaScript的内置对象,为开发者提供了更加便利的编程体验。本实例将深入探讨如何利用Prototype.js来实现弹出窗口和移动窗口的功能,这对于创建交互性强...
Prototype.js是最早期的JavaScript框架之一,它的设计目标是增强JavaScript的基本功能,使得JavaScript的面向对象编程更加简洁和强大。Prototype的核心特性包括: 1. **对象扩展**:Prototype通过扩展JavaScript的...
prototype.js是一个非常优雅的javascript基础类库,对javascript做了大量的扩展,而且很好的支持Ajax,国外有多个基于此类库实现的效果库,也做得很棒。 prototype.js不仅是一个有很大实用价值的js库,而且有很...
Prototype.js 是一个 JavaScript 库,它的主要目标是增强和扩展JavaScript的基本功能,使其更适合开发复杂的Web应用程序。在本文中,我们将深入探讨Prototype.js的核心概念、关键功能和改进,特别是集中在1.3.1版本...
在Prototype.js中,可以使用`ModalBox`插件或者自定义JavaScript代码来实现这一效果。`ModalBox`是Prototype.js的一个扩展,提供了一种创建模态对话框的简单方式。 创建模态弹出窗口的步骤如下: 1. **HTML结构**...
Prototype.js是一款强大的JavaScript库,由Sam Stephenson开发,旨在简化JavaScript的编程,提供了一系列实用的函数和类扩展,增强了JavaScript的基本功能。这款库的核心理念是通过类原型(Class Prototyping)实现...
《prototype.js 说明文档》是关于JavaScript库prototype.js的详细指南,主要涵盖了其核心概念、通用方法以及Ajax对象的使用等内容。Prototype是一个由Sam Stephenson编写的JavaScript库,旨在简化和增强JavaScript...
实践中,可以尝试用Prototype.js重构一些传统的JavaScript代码,体验其带来的便利。同时,结合实际的网页应用,比如构建动态表格、实现无刷新更新等,可以更好地掌握Prototype.js的用法。 总的来说,Prototype.js是...