2 LowPro
.Version
= '0.4.1';
4 if (!Element
.addMethods
)
5 Element
.addMethods = function(o
) { Object
.extend(Element
.Methods
, o
) };
7 // Simple utility methods for working with the DOM
9 insertAfter : function(element
, node
, otherNode
) {
11 if (otherNode
.nextSibling
)
12 return element
.insertBefore(node
, otherNode
.nextSibling
);
14 return element
.appendChild(node
);
16 addBefore : function(element
, node
) {
18 return element
.parentNode
.insertBefore(node
, element
);
20 addAfter : function(element
, node
) {
22 return $(element
.parentNode
).insertAfter(node
, element
);
24 replaceElement : function(element
, node
) {
25 $(element
).parentNode
.replaceChild(node
, element
);
28 prependChild : function(element
, node
) {
29 $(element
).insertBefore(node
, element
.firstChild
);
31 appendChildren : function(element
, children
) {
33 if (!(children
instanceof Array
))
34 children
= Array
.prototype.slice
.call(arguments
, 1);
35 children
.each(function(child
) { element
.appendChild(child
) });
40 // Add them to the element mixin
41 Element
.addMethods(DOM
);
43 // DOMBuilder for prototype
46 'class' : 'className',
50 ieAttrSet : function(attrs
, attr
, el
) {
52 if (trans
= this.IE_TRANSLATIONS
[attr
]) el
[trans
] = attrs
[attr
];
53 else if (attr
== 'style') el
.style
.cssText
= attrs
[attr
];
54 else if (attr
.match(/^on/)) el
[attr
] = new Function(attrs
[attr
]);
55 else el
.setAttribute(attr
, attrs
[attr
]);
57 getElement : function(tag
) {
58 var element
= DOM
.Builder
.cache
[tag
];
60 element
= DOM
.Builder
.cache
[tag
] = document
.createElement(tag
);
61 return element
.cloneNode(false);
63 tagFunc : function(tag
) {
66 if (arguments
.length
>0) {
67 if (arguments
[0].nodeName
||
68 typeof arguments
[0] == "string")
72 children
= Array
.prototype.slice
.call(arguments
, 1);
75 return DOM
.Builder
.create(tag
, attrs
, children
);
78 create : function(tag
, attrs
, children
) {
79 attrs
= attrs
|| {}; children
= children
|| []; tag
= tag
.toLowerCase();
80 var isIE
= navigator
.userAgent
.match(/MSIE/);
81 var el
= (isIE
&& attrs
.name
) ?
82 document
.createElement("<" + tag
+ " name=" + attrs
.name
+ ">") :
83 DOM
.Builder
.getElement(tag
);
85 for (var attr
in attrs
) {
86 if (attrs
[attr
] === true) attrs
[attr
] = attr
;
87 if (typeof attrs
[attr
] != 'function') {
88 if (isIE
) this.ieAttrSet(attrs
, attr
, el
);
89 else el
.setAttribute(attr
, attrs
[attr
].toString());
90 } else if (attr
.match(/^on(.+)$/)) {
91 Event
.observe(el
, RegExp
.$1, attrs
[attr
]);
95 for (var i
=0; i
<children
.length
; i
++) {
96 if (typeof children
[i
] == 'string')
97 children
[i
] = document
.createTextNode(children
[i
]);
98 el
.appendChild(children
[i
]);
104 // Automatically create node builders as $tagName.
106 var els
= ("p|div|span|strong|em|img|table|tr|td|th|thead|tbody|tfoot|pre|code|" +
107 "h1|h2|h3|h4|h5|h6|ul|ol|li|form|input|textarea|legend|fieldset|" +
108 "select|option|blockquote|cite|br|hr|dd|dl|dt|address|a|button|abbr|acronym|" +
109 "script|link|style|bdo|ins|del|object|param|col|colgroup|optgroup|caption|" +
110 "label|dfn|kbd|samp|var").split("|");
112 while (el
= els
[i
++])
113 window
['$' + el
] = DOM
.Builder
.tagFunc(el
);
116 DOM
.Builder
.fromHTML = function(html
) {
118 if (!(root
= arguments
.callee
._root
))
119 root
= arguments
.callee
._root
= document
.createElement('div');
120 root
.innerHTML
= html
;
121 return root
.childNodes
[0];
124 String
.prototype.toElement = function() {
125 return DOM
.Builder
.fromHTML(this);
130 $ = function(element
) {
131 if (element
&& element
.toElement
&& element
.match(/^<(.+)>$/))
132 return $(element
.toElement());
133 return old
$.apply(this, arguments
);
139 // Adapted from DOM Ready extension by Dan Webb
140 // http://www.vivabit.com/bollocks/2006/06/21/a-dom-ready-extension-for-prototype
141 // which was based on work by Matthias Miller, Dean Edwards and John Resig
145 // Event.onReady(callbackFunction);
146 Object
.extend(Event
, {
147 _domReady : function() {
148 if (arguments
.callee
.done
) return;
149 arguments
.callee
.done
= true;
151 if (Event
._timer
) clearInterval(Event
._timer
);
153 Event
._readyCallbacks
.each(function(f
) { f() });
154 Event
._readyCallbacks
= null;
157 onReady : function(f
) {
158 if (!this._readyCallbacks
) {
159 var domReady
= this._domReady
;
161 if (domReady
.done
) return f();
163 if (document
.addEventListener
)
164 document
.addEventListener("DOMContentLoaded", domReady
, false);
168 var dummy = location.protocol == "https:" ? "https://javascript:void(0)" : "javascript:void(0)";
169 document.write("<script id=__ie_onload defer src='" + dummy + "'><\/script>");
170 document.getElementById("__ie_onload").onreadystatechange = function() {
171 if (this.readyState == "complete") { domReady(); }
175 if (/WebKit/i.test(navigator
.userAgent
)) {
176 this._timer
= setInterval(function() {
177 if (/loaded|complete/.test(document
.readyState
)) domReady();
181 Event
.observe(window
, 'load', domReady
);
182 Event
._readyCallbacks
= [];
184 Event
._readyCallbacks
.push(f
);
188 // Extend Element with observe and stopObserving.
189 if (typeof Element
.Methods
.observe
== 'undefined') Element
.addMethods({
190 observe : function(el
, event
, callback
) {
191 Event
.observe(el
, event
, callback
);
193 stopObserving : function(el
, event
, callback
) {
194 Event
.stopObserving(el
, event
, callback
);
198 // Replace out existing event observe code with Dean Edwards' addEvent
199 // http://dean.edwards.name/weblog/2005/10/add-event/
200 Object
.extend(Event
, {
201 _observeAndCache : function(el
, type
, func
) {
202 if (!func
.$$guid
) func
.$$guid
= Event
._guid
++;
203 if (!el
.events
) el
.events
= {};
204 var handlers
= el
.events
[type
];
206 handlers
= el
.events
[type
] = {};
207 if (el
["on" + type
]) {
208 handlers
[0] = el
["on" + type
];
211 handlers
[func
.$$guid
] = func
;
212 el
["on" + type
] = Event
._handleEvent
;
214 if (!Event
.observers
) Event
.observers
= [];
215 Event
.observers
.push([el
, type
, func
, false]);
217 stopObserving : function(el
, type
, func
) {
219 if (el
.events
&& el
.events
[type
]) delete el
.events
[type
][func
.$$guid
];
221 for (var i
= 0; i
< Event
.observers
.length
; i
++) {
222 if (Event
.observers
[i
] &&
223 Event
.observers
[i
][0] == el
&&
224 Event
.observers
[i
][1] == type
&&
225 Event
.observers
[i
][2] == func
) delete Event
.observers
[i
];
228 _handleEvent : function(e
) {
229 var returnValue
= true;
230 e
= e
|| Event
._fixEvent(window
.event
);
231 var handlers
= this.events
[e
.type
], el
= $(this);
232 for (var i
in handlers
) {
233 el
.$$handleEvent
= handlers
[i
];
234 if (el
.$$handleEvent(e
) === false) returnValue
= false;
236 if (returnValue
== false) e
.preventDefault();
239 _fixEvent : function(e
) {
240 e
.preventDefault
= Event
._preventDefault
;
241 e
.stopPropagation
= Event
._stopPropagation
;
244 _preventDefault : function() { this.returnValue
= false },
245 _stopPropagation : function() { this.cancelBubble
= true },
249 // Allows you to trigger an event element.
250 Object
.extend(Event
, {
251 trigger : function(element
, event
, fakeEvent
) {
252 element
= $(element
);
253 fakeEvent
= fakeEvent
|| { type
: event
};
254 if(element
.events
&& element
.events
[event
]) {
255 $H(element
.events
[event
]).each(function(cache
) {
256 cache
[1].call(element
, fakeEvent
);
262 // Based on event:Selectors by Justin Palmer
263 // http://encytemedia.com/event-selectors/
267 // Event.addBehavior({
268 // "selector:event" : function(event) { /* event handler. this refers to the element. */ },
269 // "selector" : function() { /* runs function on dom ready. this refers to the element. */ }
273 // Multiple calls will add to exisiting rules. Event.addBehavior.reassignAfterAjax and
274 // Event.addBehavior.autoTrigger can be adjusted to needs.
275 Event
.addBehavior = function(rules
) {
276 var ab
= this.addBehavior
;
277 Object
.extend(ab
.rules
, rules
);
279 if (!ab
.responderApplied
) {
280 Ajax
.Responders
.register({
281 onComplete : function() {
282 if (Event
.addBehavior
.reassignAfterAjax
)
283 setTimeout(function() { ab
.unload(); ab
.load(ab
.rules
) }, 10);
286 ab
.responderApplied
= true;
289 if (ab
.autoTrigger
) {
290 this.onReady(ab
.load
.bind(ab
, rules
));
295 Object
.extend(Event
.addBehavior
, {
296 rules
: {}, cache
: [],
297 reassignAfterAjax
: true,
300 load : function(rules
) {
301 for (var selector
in rules
) {
302 var observer
= rules
[selector
];
303 var sels
= selector
.split(',');
304 sels
.each(function(sel
) {
305 var parts
= sel
.split(/:(?=[a-z]+$)/), css
= parts
[0], event
= parts
[1];
306 $$(css
).each(function(element
) {
308 $(element
).observe(event
, observer
);
309 Event
.addBehavior
.cache
.push([element
, event
, observer
]);
311 if (!element
.$$assigned
|| !element
.$$assigned
.include(observer
)) {
312 if (observer
.attach
) observer
.attach(element
);
314 else observer
.call($(element
));
315 element
.$$assigned
= element
.$$assigned
|| [];
316 element
.$$assigned
.push(observer
);
324 unload : function() {
325 this.cache
.each(function(c
) {
326 Event
.stopObserving
.apply(Event
, c
);
333 Event
.observe(window
, 'unload', Event
.addBehavior
.unload
.bind(Event
.addBehavior
));
335 // A silly Prototype style shortcut for the reckless
336 $$$ = Event
.addBehavior
;
338 // Behaviors can be bound to elements to provide an object orientated way of controlling elements
339 // and their behavior. Use Behavior.create() to make a new behavior class then use attach() to
340 // glue it to an element. Each element then gets it's own instance of the behavior and any
341 // methods called onxxx are bound to the relevent event.
345 // var MyBehavior = Behavior.create({
346 // onmouseover : function() { this.element.addClassName('bong') }
349 // Event.addBehavior({ 'a.rollover' : MyBehavior });
351 // If you need to pass additional values to initialize use:
353 // Event.addBehavior({ 'a.rollover' : MyBehavior(10, { thing : 15 }) })
355 // You can also use the attach() method. If you specify extra arguments to attach they get passed to initialize.
357 // MyBehavior.attach(el, values, to, init);
359 // Finally, the rawest method is using the new constructor normally:
360 // var draggable = new Draggable(element, init, vals);
362 // Each behaviour has a collection of all its instances in Behavior.instances
365 create : function(members
) {
366 var behavior = function() {
367 var behavior
= arguments
.callee
;
368 if (this == window
|| $H(this).values().include(behavior
)) {
370 for (var i
= 0; i
< arguments
.length
; i
++)
371 args
.push(arguments
[i
]);
374 var initArgs
= [this].concat(args
);
375 behavior
.attach
.apply(behavior
, initArgs
);
378 var args
= (arguments
.length
== 2 && arguments
[1] instanceof Array
) ?
379 arguments
[1] : Array
.prototype.slice
.call(arguments
, 1);
381 this.element
= $(arguments
[0]);
382 this.initialize
.apply(this, args
);
383 behavior
._bindEvents(this);
384 behavior
.instances
.push(this);
387 behavior
.prototype.initialize
= Prototype
.K
;
388 Object
.extend(behavior
.prototype, members
);
389 Object
.extend(behavior
, Behavior
.ClassMethods
);
390 behavior
.instances
= [];
394 attach : function(element
) {
395 return new this(element
, Array
.prototype.slice
.call(arguments
, 1));
397 _bindEvents : function(bound
) {
398 for (var member
in bound
)
399 if (member
.match(/^on(.+)/) && typeof bound
[member
] == 'function')
400 bound
.element
.observe(RegExp
.$1, bound
[member
].bindAsEventListener(bound
));