2 // Technique from Juriy Zaytsev
3 // http://thinkweb2.com/projects/prototype/detecting-event-support-without-browser-sniffing/
4 function isEventSupported(eventName) {
5 var el = document.createElement('div');
6 eventName = 'on' + eventName;
7 var isSupported = (eventName in el);
9 el.setAttribute(eventName, 'return;');
10 isSupported = typeof el[eventName] == 'function';
16 function isForm(element) {
17 return Object.isElement(element) && element.nodeName.toUpperCase() == 'FORM'
20 function isInput(element) {
21 if (Object.isElement(element)) {
22 var name = element.nodeName.toUpperCase()
23 return name == 'INPUT' || name == 'SELECT' || name == 'TEXTAREA'
28 var submitBubbles = isEventSupported('submit'),
29 changeBubbles = isEventSupported('change')
31 if (!submitBubbles || !changeBubbles) {
32 // augment the Event.Handler class to observe custom events when needed
33 Event.Handler.prototype.initialize = Event.Handler.prototype.initialize.wrap(
34 function(init, element, eventName, selector, callback) {
35 init(element, eventName, selector, callback)
36 // is the handler being attached to an element that doesn't support this event?
37 if ( (!submitBubbles && this.eventName == 'submit' && !isForm(this.element)) ||
38 (!changeBubbles && this.eventName == 'change' && !isInput(this.element)) ) {
39 // "submit" => "emulated:submit"
40 this.eventName = 'emulated:' + this.eventName
47 // discover forms on the page by observing focus events which always bubble
48 document.on('focusin', 'form', function(focusEvent, form) {
49 // special handler for the real "submit" event (one-time operation)
50 if (!form.retrieve('emulated:submit')) {
51 form.on('submit', function(submitEvent) {
52 var emulated = form.fire('emulated:submit', submitEvent, true)
53 // if custom event received preventDefault, cancel the real one too
54 if (emulated.returnValue === false) submitEvent.preventDefault()
56 form.store('emulated:submit', true)
62 // discover form inputs on the page
63 document.on('focusin', 'input, select, texarea', function(focusEvent, input) {
64 // special handler for real "change" events
65 if (!input.retrieve('emulated:change')) {
66 input.on('change', function(changeEvent) {
67 input.fire('emulated:change', changeEvent, true)
69 input.store('emulated:change', true)
74 function handleRemote(element) {
75 var method, url, params;
77 var event = element.fire("ajax:before");
78 if (event.stopped) return false;
80 if (element.tagName.toLowerCase() === 'form') {
81 method = element.readAttribute('method') || 'post';
82 url = element.readAttribute('action');
83 params = element.serialize();
85 method = element.readAttribute('data-method') || 'get';
86 url = element.readAttribute('href');
90 new Ajax.Request(url, {
95 onComplete: function(request) { element.fire("ajax:complete", request); },
96 onSuccess: function(request) { element.fire("ajax:success", request); },
97 onFailure: function(request) { element.fire("ajax:failure", request); }
100 element.fire("ajax:after");
103 function handleMethod(element) {
104 var method = element.readAttribute('data-method'),
105 url = element.readAttribute('href'),
106 csrf_param = $$('meta[name=csrf-param]')[0],
107 csrf_token = $$('meta[name=csrf-token]')[0];
109 var form = new Element('form', { method: "POST", action: url, style: "display: none;" });
110 element.parentNode.insert(form);
112 if (method !== 'post') {
113 var field = new Element('input', { type: 'hidden', name: '_method', value: method });
118 var param = csrf_param.readAttribute('content'),
119 token = csrf_token.readAttribute('content'),
120 field = new Element('input', { type: 'hidden', name: param, value: token });
128 document.on("click", "*[data-confirm]", function(event, element) {
129 var message = element.readAttribute('data-confirm');
130 if (!confirm(message)) event.stop();
133 document.on("click", "a[data-remote]", function(event, element) {
134 if (event.stopped) return;
135 handleRemote(element);
139 document.on("click", "a[data-method]", function(event, element) {
140 if (event.stopped) return;
141 handleMethod(element);
145 document.on("submit", function(event) {
146 var element = event.findElement(),
147 message = element.readAttribute('data-confirm');
148 if (message && !confirm(message)) {
153 var inputs = element.select("input[type=submit][data-disable-with]");
154 inputs.each(function(input) {
155 input.disabled = true;
156 input.writeAttribute('data-original-value', input.value);
157 input.value = input.readAttribute('data-disable-with');
160 var element = event.findElement("form[data-remote]");
162 handleRemote(element);
167 document.on("ajax:after", "form", function(event, element) {
168 var inputs = element.select("input[type=submit][disabled=true][data-disable-with]");
169 inputs.each(function(input) {
170 input.value = input.readAttribute('data-original-value');
171 input.removeAttribute('data-original-value');
172 input.disabled = false;
176 Ajax.Responders.register({
177 onCreate: function(request) {
178 var csrf_meta_tag = $$('meta[name=csrf-token]')[0];
181 var header = 'X-CSRF-Token',
182 token = csrf_meta_tag.readAttribute('content');
184 if (!request.options.requestHeaders) {
185 request.options.requestHeaders = {};
187 request.options.requestHeaders[header] = token;