user_query 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (69) hide show
  1. data/ChangeLog +4 -0
  2. data/README +45 -0
  3. data/Rakefile +359 -0
  4. data/Releases +6 -0
  5. data/TODO +0 -0
  6. data/examples/userqueryex/HOWTO.txt +5 -0
  7. data/examples/userqueryex/README +183 -0
  8. data/examples/userqueryex/Rakefile +10 -0
  9. data/examples/userqueryex/WHAT.txt +16 -0
  10. data/examples/userqueryex/app/controllers/application.rb +4 -0
  11. data/examples/userqueryex/app/controllers/entries_controller.rb +68 -0
  12. data/examples/userqueryex/app/helpers/application_helper.rb +3 -0
  13. data/examples/userqueryex/app/helpers/entries_helper.rb +2 -0
  14. data/examples/userqueryex/app/models/entry.rb +8 -0
  15. data/examples/userqueryex/app/views/entries/_form.rhtml +20 -0
  16. data/examples/userqueryex/app/views/entries/edit.rhtml +9 -0
  17. data/examples/userqueryex/app/views/entries/list.rhtml +75 -0
  18. data/examples/userqueryex/app/views/entries/new.rhtml +8 -0
  19. data/examples/userqueryex/app/views/entries/show.rhtml +8 -0
  20. data/examples/userqueryex/app/views/layouts/entries.rhtml +13 -0
  21. data/examples/userqueryex/config/boot.rb +44 -0
  22. data/examples/userqueryex/config/database.yml +36 -0
  23. data/examples/userqueryex/config/environment.rb +54 -0
  24. data/examples/userqueryex/config/environments/development.rb +21 -0
  25. data/examples/userqueryex/config/environments/production.rb +18 -0
  26. data/examples/userqueryex/config/environments/test.rb +19 -0
  27. data/examples/userqueryex/config/routes.rb +22 -0
  28. data/examples/userqueryex/db/migrate/001_entry_migration.rb +16 -0
  29. data/examples/userqueryex/db/schema.rb +15 -0
  30. data/examples/userqueryex/doc/README_FOR_APP +2 -0
  31. data/examples/userqueryex/public/404.html +8 -0
  32. data/examples/userqueryex/public/500.html +8 -0
  33. data/examples/userqueryex/public/dispatch.cgi +10 -0
  34. data/examples/userqueryex/public/dispatch.fcgi +24 -0
  35. data/examples/userqueryex/public/dispatch.rb +10 -0
  36. data/examples/userqueryex/public/favicon.ico +0 -0
  37. data/examples/userqueryex/public/images/rails.png +0 -0
  38. data/examples/userqueryex/public/javascripts/application.js +2 -0
  39. data/examples/userqueryex/public/javascripts/controls.js +815 -0
  40. data/examples/userqueryex/public/javascripts/dragdrop.js +913 -0
  41. data/examples/userqueryex/public/javascripts/effects.js +958 -0
  42. data/examples/userqueryex/public/javascripts/prototype.js +2006 -0
  43. data/examples/userqueryex/public/robots.txt +1 -0
  44. data/examples/userqueryex/public/stylesheets/scaffold.css +74 -0
  45. data/examples/userqueryex/script/about +3 -0
  46. data/examples/userqueryex/script/breakpointer +3 -0
  47. data/examples/userqueryex/script/console +3 -0
  48. data/examples/userqueryex/script/destroy +3 -0
  49. data/examples/userqueryex/script/generate +3 -0
  50. data/examples/userqueryex/script/performance/benchmarker +3 -0
  51. data/examples/userqueryex/script/performance/profiler +3 -0
  52. data/examples/userqueryex/script/plugin +3 -0
  53. data/examples/userqueryex/script/process/reaper +3 -0
  54. data/examples/userqueryex/script/process/spawner +3 -0
  55. data/examples/userqueryex/script/runner +3 -0
  56. data/examples/userqueryex/script/server +3 -0
  57. data/examples/userqueryex/test/fixtures/entries.yml +5 -0
  58. data/examples/userqueryex/test/functional/entries_controller_test.rb +88 -0
  59. data/examples/userqueryex/test/test_helper.rb +28 -0
  60. data/examples/userqueryex/test/unit/entry_test.rb +10 -0
  61. data/lib/user_query.rb +10 -0
  62. data/lib/user_query/generator.rb +219 -0
  63. data/lib/user_query/parameters.rb +93 -0
  64. data/lib/user_query/parser.rb +762 -0
  65. data/lib/user_query/schema.rb +159 -0
  66. data/lib/user_query/user_query_version.rb +6 -0
  67. data/test/parser_test.rb +539 -0
  68. data/test/schema_test.rb +142 -0
  69. metadata +148 -0
@@ -0,0 +1,19 @@
1
+ # Settings specified here will take precedence over those in config/environment.rb
2
+
3
+ # The test environment is used exclusively to run your application's
4
+ # test suite. You never need to work with it otherwise. Remember that
5
+ # your test database is "scratch space" for the test suite and is wiped
6
+ # and recreated between test runs. Don't rely on the data there!
7
+ config.cache_classes = true
8
+
9
+ # Log error messages when you accidentally call methods on nil.
10
+ config.whiny_nils = true
11
+
12
+ # Show full error reports and disable caching
13
+ config.action_controller.consider_all_requests_local = true
14
+ config.action_controller.perform_caching = false
15
+
16
+ # Tell ActionMailer not to deliver emails to the real world.
17
+ # The :test delivery method accumulates sent emails in the
18
+ # ActionMailer::Base.deliveries array.
19
+ config.action_mailer.delivery_method = :test
@@ -0,0 +1,22 @@
1
+ ActionController::Routing::Routes.draw do |map|
2
+ # The priority is based upon order of creation: first created -> highest priority.
3
+
4
+ # Sample of regular route:
5
+ # map.connect 'products/:id', :controller => 'catalog', :action => 'view'
6
+ # Keep in mind you can assign values other than :controller and :action
7
+
8
+ # Sample of named route:
9
+ # map.purchase 'products/:id/purchase', :controller => 'catalog', :action => 'purchase'
10
+ # This route can be invoked with purchase_url(:id => product.id)
11
+
12
+ # You can have the root of your site routed by hooking up ''
13
+ # -- just remember to delete public/index.html.
14
+ map.connect '', :controller => "entries"
15
+
16
+ # Allow downloading Web Service WSDL as a file with an extension
17
+ # instead of a file named 'wsdl'
18
+ map.connect ':controller/service.wsdl', :action => 'wsdl'
19
+
20
+ # Install the default route as the lowest priority.
21
+ map.connect ':controller/:action/:id'
22
+ end
@@ -0,0 +1,16 @@
1
+ # $Id$
2
+ class EntryMigration < ActiveRecord::Migration
3
+ def self.up
4
+ create_table "entries", :force => true do |t|
5
+ t.column "name", :string
6
+ t.column "date", :datetime
7
+ t.column "memo", :text
8
+ t.column "amount", :integer # Currency::Money
9
+ t.column "approved", :boolean
10
+ end
11
+ end
12
+
13
+ def self.down
14
+ drop_table "entries"
15
+ end
16
+ end
@@ -0,0 +1,15 @@
1
+ # This file is autogenerated. Instead of editing this file, please use the
2
+ # migrations feature of ActiveRecord to incrementally modify your database, and
3
+ # then regenerate this schema definition.
4
+
5
+ ActiveRecord::Schema.define(:version => 1) do
6
+
7
+ create_table "entries", :force => true do |t|
8
+ t.column "name", :string
9
+ t.column "date", :datetime
10
+ t.column "memo", :text
11
+ t.column "amount", :integer
12
+ t.column "approved", :boolean
13
+ end
14
+
15
+ end
@@ -0,0 +1,2 @@
1
+ Use this README file to introduce your application and point to useful places in the API for learning more.
2
+ Run "rake appdoc" to generate API documentation for your models and controllers.
@@ -0,0 +1,8 @@
1
+ <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
2
+ "http://www.w3.org/TR/html4/loose.dtd">
3
+ <html>
4
+ <body>
5
+ <h1>File not found</h1>
6
+ <p>Change this error message for pages not found in public/404.html</p>
7
+ </body>
8
+ </html>
@@ -0,0 +1,8 @@
1
+ <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
2
+ "http://www.w3.org/TR/html4/loose.dtd">
3
+ <html>
4
+ <body>
5
+ <h1>Application error</h1>
6
+ <p>Change this error message for exceptions thrown outside of an action (like in Dispatcher setups or broken Ruby code) in public/500.html</p>
7
+ </body>
8
+ </html>
@@ -0,0 +1,10 @@
1
+ #!/usr/bin/ruby
2
+
3
+ require File.dirname(__FILE__) + "/../config/environment" unless defined?(RAILS_ROOT)
4
+
5
+ # If you're using RubyGems and mod_ruby, this require should be changed to an absolute path one, like:
6
+ # "/usr/local/lib/ruby/gems/1.8/gems/rails-0.8.0/lib/dispatcher" -- otherwise performance is severely impaired
7
+ require "dispatcher"
8
+
9
+ ADDITIONAL_LOAD_PATHS.reverse.each { |dir| $:.unshift(dir) if File.directory?(dir) } if defined?(Apache::RubyRun)
10
+ Dispatcher.dispatch
@@ -0,0 +1,24 @@
1
+ #!/usr/bin/ruby
2
+ #
3
+ # You may specify the path to the FastCGI crash log (a log of unhandled
4
+ # exceptions which forced the FastCGI instance to exit, great for debugging)
5
+ # and the number of requests to process before running garbage collection.
6
+ #
7
+ # By default, the FastCGI crash log is RAILS_ROOT/log/fastcgi.crash.log
8
+ # and the GC period is nil (turned off). A reasonable number of requests
9
+ # could range from 10-100 depending on the memory footprint of your app.
10
+ #
11
+ # Example:
12
+ # # Default log path, normal GC behavior.
13
+ # RailsFCGIHandler.process!
14
+ #
15
+ # # Default log path, 50 requests between GC.
16
+ # RailsFCGIHandler.process! nil, 50
17
+ #
18
+ # # Custom log path, normal GC behavior.
19
+ # RailsFCGIHandler.process! '/var/log/myapp_fcgi_crash.log'
20
+ #
21
+ require File.dirname(__FILE__) + "/../config/environment"
22
+ require 'fcgi_handler'
23
+
24
+ RailsFCGIHandler.process!
@@ -0,0 +1,10 @@
1
+ #!/usr/bin/ruby
2
+
3
+ require File.dirname(__FILE__) + "/../config/environment" unless defined?(RAILS_ROOT)
4
+
5
+ # If you're using RubyGems and mod_ruby, this require should be changed to an absolute path one, like:
6
+ # "/usr/local/lib/ruby/gems/1.8/gems/rails-0.8.0/lib/dispatcher" -- otherwise performance is severely impaired
7
+ require "dispatcher"
8
+
9
+ ADDITIONAL_LOAD_PATHS.reverse.each { |dir| $:.unshift(dir) if File.directory?(dir) } if defined?(Apache::RubyRun)
10
+ Dispatcher.dispatch
@@ -0,0 +1,2 @@
1
+ // Place your application-specific JavaScript functions and classes here
2
+ // This file is automatically included by javascript_include_tag :defaults
@@ -0,0 +1,815 @@
1
+ // Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
2
+ // (c) 2005 Ivan Krstic (http://blogs.law.harvard.edu/ivan)
3
+ // (c) 2005 Jon Tirsen (http://www.tirsen.com)
4
+ // Contributors:
5
+ // Richard Livsey
6
+ // Rahul Bhargava
7
+ // Rob Wills
8
+ //
9
+ // See scriptaculous.js for full license.
10
+
11
+ // Autocompleter.Base handles all the autocompletion functionality
12
+ // that's independent of the data source for autocompletion. This
13
+ // includes drawing the autocompletion menu, observing keyboard
14
+ // and mouse events, and similar.
15
+ //
16
+ // Specific autocompleters need to provide, at the very least,
17
+ // a getUpdatedChoices function that will be invoked every time
18
+ // the text inside the monitored textbox changes. This method
19
+ // should get the text for which to provide autocompletion by
20
+ // invoking this.getToken(), NOT by directly accessing
21
+ // this.element.value. This is to allow incremental tokenized
22
+ // autocompletion. Specific auto-completion logic (AJAX, etc)
23
+ // belongs in getUpdatedChoices.
24
+ //
25
+ // Tokenized incremental autocompletion is enabled automatically
26
+ // when an autocompleter is instantiated with the 'tokens' option
27
+ // in the options parameter, e.g.:
28
+ // new Ajax.Autocompleter('id','upd', '/url/', { tokens: ',' });
29
+ // will incrementally autocomplete with a comma as the token.
30
+ // Additionally, ',' in the above example can be replaced with
31
+ // a token array, e.g. { tokens: [',', '\n'] } which
32
+ // enables autocompletion on multiple tokens. This is most
33
+ // useful when one of the tokens is \n (a newline), as it
34
+ // allows smart autocompletion after linebreaks.
35
+
36
+ var Autocompleter = {}
37
+ Autocompleter.Base = function() {};
38
+ Autocompleter.Base.prototype = {
39
+ baseInitialize: function(element, update, options) {
40
+ this.element = $(element);
41
+ this.update = $(update);
42
+ this.hasFocus = false;
43
+ this.changed = false;
44
+ this.active = false;
45
+ this.index = 0;
46
+ this.entryCount = 0;
47
+
48
+ if (this.setOptions)
49
+ this.setOptions(options);
50
+ else
51
+ this.options = options || {};
52
+
53
+ this.options.paramName = this.options.paramName || this.element.name;
54
+ this.options.tokens = this.options.tokens || [];
55
+ this.options.frequency = this.options.frequency || 0.4;
56
+ this.options.minChars = this.options.minChars || 1;
57
+ this.options.onShow = this.options.onShow ||
58
+ function(element, update){
59
+ if(!update.style.position || update.style.position=='absolute') {
60
+ update.style.position = 'absolute';
61
+ Position.clone(element, update, {setHeight: false, offsetTop: element.offsetHeight});
62
+ }
63
+ Effect.Appear(update,{duration:0.15});
64
+ };
65
+ this.options.onHide = this.options.onHide ||
66
+ function(element, update){ new Effect.Fade(update,{duration:0.15}) };
67
+
68
+ if (typeof(this.options.tokens) == 'string')
69
+ this.options.tokens = new Array(this.options.tokens);
70
+
71
+ this.observer = null;
72
+
73
+ this.element.setAttribute('autocomplete','off');
74
+
75
+ Element.hide(this.update);
76
+
77
+ Event.observe(this.element, "blur", this.onBlur.bindAsEventListener(this));
78
+ Event.observe(this.element, "keypress", this.onKeyPress.bindAsEventListener(this));
79
+ },
80
+
81
+ show: function() {
82
+ if(Element.getStyle(this.update, 'display')=='none') this.options.onShow(this.element, this.update);
83
+ if(!this.iefix &&
84
+ (navigator.appVersion.indexOf('MSIE')>0) &&
85
+ (navigator.userAgent.indexOf('Opera')<0) &&
86
+ (Element.getStyle(this.update, 'position')=='absolute')) {
87
+ new Insertion.After(this.update,
88
+ '<iframe id="' + this.update.id + '_iefix" '+
89
+ 'style="display:none;position:absolute;filter:progid:DXImageTransform.Microsoft.Alpha(opacity=0);" ' +
90
+ 'src="javascript:false;" frameborder="0" scrolling="no"></iframe>');
91
+ this.iefix = $(this.update.id+'_iefix');
92
+ }
93
+ if(this.iefix) setTimeout(this.fixIEOverlapping.bind(this), 50);
94
+ },
95
+
96
+ fixIEOverlapping: function() {
97
+ Position.clone(this.update, this.iefix);
98
+ this.iefix.style.zIndex = 1;
99
+ this.update.style.zIndex = 2;
100
+ Element.show(this.iefix);
101
+ },
102
+
103
+ hide: function() {
104
+ this.stopIndicator();
105
+ if(Element.getStyle(this.update, 'display')!='none') this.options.onHide(this.element, this.update);
106
+ if(this.iefix) Element.hide(this.iefix);
107
+ },
108
+
109
+ startIndicator: function() {
110
+ if(this.options.indicator) Element.show(this.options.indicator);
111
+ },
112
+
113
+ stopIndicator: function() {
114
+ if(this.options.indicator) Element.hide(this.options.indicator);
115
+ },
116
+
117
+ onKeyPress: function(event) {
118
+ if(this.active)
119
+ switch(event.keyCode) {
120
+ case Event.KEY_TAB:
121
+ case Event.KEY_RETURN:
122
+ this.selectEntry();
123
+ Event.stop(event);
124
+ case Event.KEY_ESC:
125
+ this.hide();
126
+ this.active = false;
127
+ Event.stop(event);
128
+ return;
129
+ case Event.KEY_LEFT:
130
+ case Event.KEY_RIGHT:
131
+ return;
132
+ case Event.KEY_UP:
133
+ this.markPrevious();
134
+ this.render();
135
+ if(navigator.appVersion.indexOf('AppleWebKit')>0) Event.stop(event);
136
+ return;
137
+ case Event.KEY_DOWN:
138
+ this.markNext();
139
+ this.render();
140
+ if(navigator.appVersion.indexOf('AppleWebKit')>0) Event.stop(event);
141
+ return;
142
+ }
143
+ else
144
+ if(event.keyCode==Event.KEY_TAB || event.keyCode==Event.KEY_RETURN ||
145
+ (navigator.appVersion.indexOf('AppleWebKit') > 0 && event.keyCode == 0)) return;
146
+
147
+ this.changed = true;
148
+ this.hasFocus = true;
149
+
150
+ if(this.observer) clearTimeout(this.observer);
151
+ this.observer =
152
+ setTimeout(this.onObserverEvent.bind(this), this.options.frequency*1000);
153
+ },
154
+
155
+ activate: function() {
156
+ this.changed = false;
157
+ this.hasFocus = true;
158
+ this.getUpdatedChoices();
159
+ },
160
+
161
+ onHover: function(event) {
162
+ var element = Event.findElement(event, 'LI');
163
+ if(this.index != element.autocompleteIndex)
164
+ {
165
+ this.index = element.autocompleteIndex;
166
+ this.render();
167
+ }
168
+ Event.stop(event);
169
+ },
170
+
171
+ onClick: function(event) {
172
+ var element = Event.findElement(event, 'LI');
173
+ this.index = element.autocompleteIndex;
174
+ this.selectEntry();
175
+ this.hide();
176
+ },
177
+
178
+ onBlur: function(event) {
179
+ // needed to make click events working
180
+ setTimeout(this.hide.bind(this), 250);
181
+ this.hasFocus = false;
182
+ this.active = false;
183
+ },
184
+
185
+ render: function() {
186
+ if(this.entryCount > 0) {
187
+ for (var i = 0; i < this.entryCount; i++)
188
+ this.index==i ?
189
+ Element.addClassName(this.getEntry(i),"selected") :
190
+ Element.removeClassName(this.getEntry(i),"selected");
191
+
192
+ if(this.hasFocus) {
193
+ this.show();
194
+ this.active = true;
195
+ }
196
+ } else {
197
+ this.active = false;
198
+ this.hide();
199
+ }
200
+ },
201
+
202
+ markPrevious: function() {
203
+ if(this.index > 0) this.index--
204
+ else this.index = this.entryCount-1;
205
+ },
206
+
207
+ markNext: function() {
208
+ if(this.index < this.entryCount-1) this.index++
209
+ else this.index = 0;
210
+ },
211
+
212
+ getEntry: function(index) {
213
+ return this.update.firstChild.childNodes[index];
214
+ },
215
+
216
+ getCurrentEntry: function() {
217
+ return this.getEntry(this.index);
218
+ },
219
+
220
+ selectEntry: function() {
221
+ this.active = false;
222
+ this.updateElement(this.getCurrentEntry());
223
+ },
224
+
225
+ updateElement: function(selectedElement) {
226
+ if (this.options.updateElement) {
227
+ this.options.updateElement(selectedElement);
228
+ return;
229
+ }
230
+ var value = '';
231
+ if (this.options.select) {
232
+ var nodes = document.getElementsByClassName(this.options.select, selectedElement) || [];
233
+ if(nodes.length>0) value = Element.collectTextNodes(nodes[0], this.options.select);
234
+ } else
235
+ value = Element.collectTextNodesIgnoreClass(selectedElement, 'informal');
236
+
237
+ var lastTokenPos = this.findLastToken();
238
+ if (lastTokenPos != -1) {
239
+ var newValue = this.element.value.substr(0, lastTokenPos + 1);
240
+ var whitespace = this.element.value.substr(lastTokenPos + 1).match(/^\s+/);
241
+ if (whitespace)
242
+ newValue += whitespace[0];
243
+ this.element.value = newValue + value;
244
+ } else {
245
+ this.element.value = value;
246
+ }
247
+ this.element.focus();
248
+
249
+ if (this.options.afterUpdateElement)
250
+ this.options.afterUpdateElement(this.element, selectedElement);
251
+ },
252
+
253
+ updateChoices: function(choices) {
254
+ if(!this.changed && this.hasFocus) {
255
+ this.update.innerHTML = choices;
256
+ Element.cleanWhitespace(this.update);
257
+ Element.cleanWhitespace(this.update.firstChild);
258
+
259
+ if(this.update.firstChild && this.update.firstChild.childNodes) {
260
+ this.entryCount =
261
+ this.update.firstChild.childNodes.length;
262
+ for (var i = 0; i < this.entryCount; i++) {
263
+ var entry = this.getEntry(i);
264
+ entry.autocompleteIndex = i;
265
+ this.addObservers(entry);
266
+ }
267
+ } else {
268
+ this.entryCount = 0;
269
+ }
270
+
271
+ this.stopIndicator();
272
+
273
+ this.index = 0;
274
+ this.render();
275
+ }
276
+ },
277
+
278
+ addObservers: function(element) {
279
+ Event.observe(element, "mouseover", this.onHover.bindAsEventListener(this));
280
+ Event.observe(element, "click", this.onClick.bindAsEventListener(this));
281
+ },
282
+
283
+ onObserverEvent: function() {
284
+ this.changed = false;
285
+ if(this.getToken().length>=this.options.minChars) {
286
+ this.startIndicator();
287
+ this.getUpdatedChoices();
288
+ } else {
289
+ this.active = false;
290
+ this.hide();
291
+ }
292
+ },
293
+
294
+ getToken: function() {
295
+ var tokenPos = this.findLastToken();
296
+ if (tokenPos != -1)
297
+ var ret = this.element.value.substr(tokenPos + 1).replace(/^\s+/,'').replace(/\s+$/,'');
298
+ else
299
+ var ret = this.element.value;
300
+
301
+ return /\n/.test(ret) ? '' : ret;
302
+ },
303
+
304
+ findLastToken: function() {
305
+ var lastTokenPos = -1;
306
+
307
+ for (var i=0; i<this.options.tokens.length; i++) {
308
+ var thisTokenPos = this.element.value.lastIndexOf(this.options.tokens[i]);
309
+ if (thisTokenPos > lastTokenPos)
310
+ lastTokenPos = thisTokenPos;
311
+ }
312
+ return lastTokenPos;
313
+ }
314
+ }
315
+
316
+ Ajax.Autocompleter = Class.create();
317
+ Object.extend(Object.extend(Ajax.Autocompleter.prototype, Autocompleter.Base.prototype), {
318
+ initialize: function(element, update, url, options) {
319
+ this.baseInitialize(element, update, options);
320
+ this.options.asynchronous = true;
321
+ this.options.onComplete = this.onComplete.bind(this);
322
+ this.options.defaultParams = this.options.parameters || null;
323
+ this.url = url;
324
+ },
325
+
326
+ getUpdatedChoices: function() {
327
+ entry = encodeURIComponent(this.options.paramName) + '=' +
328
+ encodeURIComponent(this.getToken());
329
+
330
+ this.options.parameters = this.options.callback ?
331
+ this.options.callback(this.element, entry) : entry;
332
+
333
+ if(this.options.defaultParams)
334
+ this.options.parameters += '&' + this.options.defaultParams;
335
+
336
+ new Ajax.Request(this.url, this.options);
337
+ },
338
+
339
+ onComplete: function(request) {
340
+ this.updateChoices(request.responseText);
341
+ }
342
+
343
+ });
344
+
345
+ // The local array autocompleter. Used when you'd prefer to
346
+ // inject an array of autocompletion options into the page, rather
347
+ // than sending out Ajax queries, which can be quite slow sometimes.
348
+ //
349
+ // The constructor takes four parameters. The first two are, as usual,
350
+ // the id of the monitored textbox, and id of the autocompletion menu.
351
+ // The third is the array you want to autocomplete from, and the fourth
352
+ // is the options block.
353
+ //
354
+ // Extra local autocompletion options:
355
+ // - choices - How many autocompletion choices to offer
356
+ //
357
+ // - partialSearch - If false, the autocompleter will match entered
358
+ // text only at the beginning of strings in the
359
+ // autocomplete array. Defaults to true, which will
360
+ // match text at the beginning of any *word* in the
361
+ // strings in the autocomplete array. If you want to
362
+ // search anywhere in the string, additionally set
363
+ // the option fullSearch to true (default: off).
364
+ //
365
+ // - fullSsearch - Search anywhere in autocomplete array strings.
366
+ //
367
+ // - partialChars - How many characters to enter before triggering
368
+ // a partial match (unlike minChars, which defines
369
+ // how many characters are required to do any match
370
+ // at all). Defaults to 2.
371
+ //
372
+ // - ignoreCase - Whether to ignore case when autocompleting.
373
+ // Defaults to true.
374
+ //
375
+ // It's possible to pass in a custom function as the 'selector'
376
+ // option, if you prefer to write your own autocompletion logic.
377
+ // In that case, the other options above will not apply unless
378
+ // you support them.
379
+
380
+ Autocompleter.Local = Class.create();
381
+ Autocompleter.Local.prototype = Object.extend(new Autocompleter.Base(), {
382
+ initialize: function(element, update, array, options) {
383
+ this.baseInitialize(element, update, options);
384
+ this.options.array = array;
385
+ },
386
+
387
+ getUpdatedChoices: function() {
388
+ this.updateChoices(this.options.selector(this));
389
+ },
390
+
391
+ setOptions: function(options) {
392
+ this.options = Object.extend({
393
+ choices: 10,
394
+ partialSearch: true,
395
+ partialChars: 2,
396
+ ignoreCase: true,
397
+ fullSearch: false,
398
+ selector: function(instance) {
399
+ var ret = []; // Beginning matches
400
+ var partial = []; // Inside matches
401
+ var entry = instance.getToken();
402
+ var count = 0;
403
+
404
+ for (var i = 0; i < instance.options.array.length &&
405
+ ret.length < instance.options.choices ; i++) {
406
+
407
+ var elem = instance.options.array[i];
408
+ var foundPos = instance.options.ignoreCase ?
409
+ elem.toLowerCase().indexOf(entry.toLowerCase()) :
410
+ elem.indexOf(entry);
411
+
412
+ while (foundPos != -1) {
413
+ if (foundPos == 0 && elem.length != entry.length) {
414
+ ret.push("<li><strong>" + elem.substr(0, entry.length) + "</strong>" +
415
+ elem.substr(entry.length) + "</li>");
416
+ break;
417
+ } else if (entry.length >= instance.options.partialChars &&
418
+ instance.options.partialSearch && foundPos != -1) {
419
+ if (instance.options.fullSearch || /\s/.test(elem.substr(foundPos-1,1))) {
420
+ partial.push("<li>" + elem.substr(0, foundPos) + "<strong>" +
421
+ elem.substr(foundPos, entry.length) + "</strong>" + elem.substr(
422
+ foundPos + entry.length) + "</li>");
423
+ break;
424
+ }
425
+ }
426
+
427
+ foundPos = instance.options.ignoreCase ?
428
+ elem.toLowerCase().indexOf(entry.toLowerCase(), foundPos + 1) :
429
+ elem.indexOf(entry, foundPos + 1);
430
+
431
+ }
432
+ }
433
+ if (partial.length)
434
+ ret = ret.concat(partial.slice(0, instance.options.choices - ret.length))
435
+ return "<ul>" + ret.join('') + "</ul>";
436
+ }
437
+ }, options || {});
438
+ }
439
+ });
440
+
441
+ // AJAX in-place editor
442
+ //
443
+ // see documentation on http://wiki.script.aculo.us/scriptaculous/show/Ajax.InPlaceEditor
444
+
445
+ // Use this if you notice weird scrolling problems on some browsers,
446
+ // the DOM might be a bit confused when this gets called so do this
447
+ // waits 1 ms (with setTimeout) until it does the activation
448
+ Field.scrollFreeActivate = function(field) {
449
+ setTimeout(function() {
450
+ Field.activate(field);
451
+ }, 1);
452
+ }
453
+
454
+ Ajax.InPlaceEditor = Class.create();
455
+ Ajax.InPlaceEditor.defaultHighlightColor = "#FFFF99";
456
+ Ajax.InPlaceEditor.prototype = {
457
+ initialize: function(element, url, options) {
458
+ this.url = url;
459
+ this.element = $(element);
460
+
461
+ this.options = Object.extend({
462
+ okButton: true,
463
+ okText: "ok",
464
+ cancelLink: true,
465
+ cancelText: "cancel",
466
+ savingText: "Saving...",
467
+ clickToEditText: "Click to edit",
468
+ okText: "ok",
469
+ rows: 1,
470
+ onComplete: function(transport, element) {
471
+ new Effect.Highlight(element, {startcolor: this.options.highlightcolor});
472
+ },
473
+ onFailure: function(transport) {
474
+ alert("Error communicating with the server: " + transport.responseText.stripTags());
475
+ },
476
+ callback: function(form) {
477
+ return Form.serialize(form);
478
+ },
479
+ handleLineBreaks: true,
480
+ loadingText: 'Loading...',
481
+ savingClassName: 'inplaceeditor-saving',
482
+ loadingClassName: 'inplaceeditor-loading',
483
+ formClassName: 'inplaceeditor-form',
484
+ highlightcolor: Ajax.InPlaceEditor.defaultHighlightColor,
485
+ highlightendcolor: "#FFFFFF",
486
+ externalControl: null,
487
+ submitOnBlur: false,
488
+ ajaxOptions: {},
489
+ evalScripts: false
490
+ }, options || {});
491
+
492
+ if(!this.options.formId && this.element.id) {
493
+ this.options.formId = this.element.id + "-inplaceeditor";
494
+ if ($(this.options.formId)) {
495
+ // there's already a form with that name, don't specify an id
496
+ this.options.formId = null;
497
+ }
498
+ }
499
+
500
+ if (this.options.externalControl) {
501
+ this.options.externalControl = $(this.options.externalControl);
502
+ }
503
+
504
+ this.originalBackground = Element.getStyle(this.element, 'background-color');
505
+ if (!this.originalBackground) {
506
+ this.originalBackground = "transparent";
507
+ }
508
+
509
+ this.element.title = this.options.clickToEditText;
510
+
511
+ this.onclickListener = this.enterEditMode.bindAsEventListener(this);
512
+ this.mouseoverListener = this.enterHover.bindAsEventListener(this);
513
+ this.mouseoutListener = this.leaveHover.bindAsEventListener(this);
514
+ Event.observe(this.element, 'click', this.onclickListener);
515
+ Event.observe(this.element, 'mouseover', this.mouseoverListener);
516
+ Event.observe(this.element, 'mouseout', this.mouseoutListener);
517
+ if (this.options.externalControl) {
518
+ Event.observe(this.options.externalControl, 'click', this.onclickListener);
519
+ Event.observe(this.options.externalControl, 'mouseover', this.mouseoverListener);
520
+ Event.observe(this.options.externalControl, 'mouseout', this.mouseoutListener);
521
+ }
522
+ },
523
+ enterEditMode: function(evt) {
524
+ if (this.saving) return;
525
+ if (this.editing) return;
526
+ this.editing = true;
527
+ this.onEnterEditMode();
528
+ if (this.options.externalControl) {
529
+ Element.hide(this.options.externalControl);
530
+ }
531
+ Element.hide(this.element);
532
+ this.createForm();
533
+ this.element.parentNode.insertBefore(this.form, this.element);
534
+ Field.scrollFreeActivate(this.editField);
535
+ // stop the event to avoid a page refresh in Safari
536
+ if (evt) {
537
+ Event.stop(evt);
538
+ }
539
+ return false;
540
+ },
541
+ createForm: function() {
542
+ this.form = document.createElement("form");
543
+ this.form.id = this.options.formId;
544
+ Element.addClassName(this.form, this.options.formClassName)
545
+ this.form.onsubmit = this.onSubmit.bind(this);
546
+
547
+ this.createEditField();
548
+
549
+ if (this.options.textarea) {
550
+ var br = document.createElement("br");
551
+ this.form.appendChild(br);
552
+ }
553
+
554
+ if (this.options.okButton) {
555
+ okButton = document.createElement("input");
556
+ okButton.type = "submit";
557
+ okButton.value = this.options.okText;
558
+ okButton.className = 'editor_ok_button';
559
+ this.form.appendChild(okButton);
560
+ }
561
+
562
+ if (this.options.cancelLink) {
563
+ cancelLink = document.createElement("a");
564
+ cancelLink.href = "#";
565
+ cancelLink.appendChild(document.createTextNode(this.options.cancelText));
566
+ cancelLink.onclick = this.onclickCancel.bind(this);
567
+ cancelLink.className = 'editor_cancel';
568
+ this.form.appendChild(cancelLink);
569
+ }
570
+ },
571
+ hasHTMLLineBreaks: function(string) {
572
+ if (!this.options.handleLineBreaks) return false;
573
+ return string.match(/<br/i) || string.match(/<p>/i);
574
+ },
575
+ convertHTMLLineBreaks: function(string) {
576
+ return string.replace(/<br>/gi, "\n").replace(/<br\/>/gi, "\n").replace(/<\/p>/gi, "\n").replace(/<p>/gi, "");
577
+ },
578
+ createEditField: function() {
579
+ var text;
580
+ if(this.options.loadTextURL) {
581
+ text = this.options.loadingText;
582
+ } else {
583
+ text = this.getText();
584
+ }
585
+
586
+ var obj = this;
587
+
588
+ if (this.options.rows == 1 && !this.hasHTMLLineBreaks(text)) {
589
+ this.options.textarea = false;
590
+ var textField = document.createElement("input");
591
+ textField.obj = this;
592
+ textField.type = "text";
593
+ textField.name = "value";
594
+ textField.value = text;
595
+ textField.style.backgroundColor = this.options.highlightcolor;
596
+ textField.className = 'editor_field';
597
+ var size = this.options.size || this.options.cols || 0;
598
+ if (size != 0) textField.size = size;
599
+ if (this.options.submitOnBlur)
600
+ textField.onblur = this.onSubmit.bind(this);
601
+ this.editField = textField;
602
+ } else {
603
+ this.options.textarea = true;
604
+ var textArea = document.createElement("textarea");
605
+ textArea.obj = this;
606
+ textArea.name = "value";
607
+ textArea.value = this.convertHTMLLineBreaks(text);
608
+ textArea.rows = this.options.rows;
609
+ textArea.cols = this.options.cols || 40;
610
+ textArea.className = 'editor_field';
611
+ if (this.options.submitOnBlur)
612
+ textArea.onblur = this.onSubmit.bind(this);
613
+ this.editField = textArea;
614
+ }
615
+
616
+ if(this.options.loadTextURL) {
617
+ this.loadExternalText();
618
+ }
619
+ this.form.appendChild(this.editField);
620
+ },
621
+ getText: function() {
622
+ return this.element.innerHTML;
623
+ },
624
+ loadExternalText: function() {
625
+ Element.addClassName(this.form, this.options.loadingClassName);
626
+ this.editField.disabled = true;
627
+ new Ajax.Request(
628
+ this.options.loadTextURL,
629
+ Object.extend({
630
+ asynchronous: true,
631
+ onComplete: this.onLoadedExternalText.bind(this)
632
+ }, this.options.ajaxOptions)
633
+ );
634
+ },
635
+ onLoadedExternalText: function(transport) {
636
+ Element.removeClassName(this.form, this.options.loadingClassName);
637
+ this.editField.disabled = false;
638
+ this.editField.value = transport.responseText.stripTags();
639
+ },
640
+ onclickCancel: function() {
641
+ this.onComplete();
642
+ this.leaveEditMode();
643
+ return false;
644
+ },
645
+ onFailure: function(transport) {
646
+ this.options.onFailure(transport);
647
+ if (this.oldInnerHTML) {
648
+ this.element.innerHTML = this.oldInnerHTML;
649
+ this.oldInnerHTML = null;
650
+ }
651
+ return false;
652
+ },
653
+ onSubmit: function() {
654
+ // onLoading resets these so we need to save them away for the Ajax call
655
+ var form = this.form;
656
+ var value = this.editField.value;
657
+
658
+ // do this first, sometimes the ajax call returns before we get a chance to switch on Saving...
659
+ // which means this will actually switch on Saving... *after* we've left edit mode causing Saving...
660
+ // to be displayed indefinitely
661
+ this.onLoading();
662
+
663
+ if (this.options.evalScripts) {
664
+ new Ajax.Request(
665
+ this.url, Object.extend({
666
+ parameters: this.options.callback(form, value),
667
+ onComplete: this.onComplete.bind(this),
668
+ onFailure: this.onFailure.bind(this),
669
+ asynchronous:true,
670
+ evalScripts:true
671
+ }, this.options.ajaxOptions));
672
+ } else {
673
+ new Ajax.Updater(
674
+ { success: this.element,
675
+ // don't update on failure (this could be an option)
676
+ failure: null },
677
+ this.url, Object.extend({
678
+ parameters: this.options.callback(form, value),
679
+ onComplete: this.onComplete.bind(this),
680
+ onFailure: this.onFailure.bind(this)
681
+ }, this.options.ajaxOptions));
682
+ }
683
+ // stop the event to avoid a page refresh in Safari
684
+ if (arguments.length > 1) {
685
+ Event.stop(arguments[0]);
686
+ }
687
+ return false;
688
+ },
689
+ onLoading: function() {
690
+ this.saving = true;
691
+ this.removeForm();
692
+ this.leaveHover();
693
+ this.showSaving();
694
+ },
695
+ showSaving: function() {
696
+ this.oldInnerHTML = this.element.innerHTML;
697
+ this.element.innerHTML = this.options.savingText;
698
+ Element.addClassName(this.element, this.options.savingClassName);
699
+ this.element.style.backgroundColor = this.originalBackground;
700
+ Element.show(this.element);
701
+ },
702
+ removeForm: function() {
703
+ if(this.form) {
704
+ if (this.form.parentNode) Element.remove(this.form);
705
+ this.form = null;
706
+ }
707
+ },
708
+ enterHover: function() {
709
+ if (this.saving) return;
710
+ this.element.style.backgroundColor = this.options.highlightcolor;
711
+ if (this.effect) {
712
+ this.effect.cancel();
713
+ }
714
+ Element.addClassName(this.element, this.options.hoverClassName)
715
+ },
716
+ leaveHover: function() {
717
+ if (this.options.backgroundColor) {
718
+ this.element.style.backgroundColor = this.oldBackground;
719
+ }
720
+ Element.removeClassName(this.element, this.options.hoverClassName)
721
+ if (this.saving) return;
722
+ this.effect = new Effect.Highlight(this.element, {
723
+ startcolor: this.options.highlightcolor,
724
+ endcolor: this.options.highlightendcolor,
725
+ restorecolor: this.originalBackground
726
+ });
727
+ },
728
+ leaveEditMode: function() {
729
+ Element.removeClassName(this.element, this.options.savingClassName);
730
+ this.removeForm();
731
+ this.leaveHover();
732
+ this.element.style.backgroundColor = this.originalBackground;
733
+ Element.show(this.element);
734
+ if (this.options.externalControl) {
735
+ Element.show(this.options.externalControl);
736
+ }
737
+ this.editing = false;
738
+ this.saving = false;
739
+ this.oldInnerHTML = null;
740
+ this.onLeaveEditMode();
741
+ },
742
+ onComplete: function(transport) {
743
+ this.leaveEditMode();
744
+ this.options.onComplete.bind(this)(transport, this.element);
745
+ },
746
+ onEnterEditMode: function() {},
747
+ onLeaveEditMode: function() {},
748
+ dispose: function() {
749
+ if (this.oldInnerHTML) {
750
+ this.element.innerHTML = this.oldInnerHTML;
751
+ }
752
+ this.leaveEditMode();
753
+ Event.stopObserving(this.element, 'click', this.onclickListener);
754
+ Event.stopObserving(this.element, 'mouseover', this.mouseoverListener);
755
+ Event.stopObserving(this.element, 'mouseout', this.mouseoutListener);
756
+ if (this.options.externalControl) {
757
+ Event.stopObserving(this.options.externalControl, 'click', this.onclickListener);
758
+ Event.stopObserving(this.options.externalControl, 'mouseover', this.mouseoverListener);
759
+ Event.stopObserving(this.options.externalControl, 'mouseout', this.mouseoutListener);
760
+ }
761
+ }
762
+ };
763
+
764
+ Ajax.InPlaceCollectionEditor = Class.create();
765
+ Object.extend(Ajax.InPlaceCollectionEditor.prototype, Ajax.InPlaceEditor.prototype);
766
+ Object.extend(Ajax.InPlaceCollectionEditor.prototype, {
767
+ createEditField: function() {
768
+ if (!this.cached_selectTag) {
769
+ var selectTag = document.createElement("select");
770
+ var collection = this.options.collection || [];
771
+ var optionTag;
772
+ collection.each(function(e,i) {
773
+ optionTag = document.createElement("option");
774
+ optionTag.value = (e instanceof Array) ? e[0] : e;
775
+ if(this.options.value==optionTag.value) optionTag.selected = true;
776
+ optionTag.appendChild(document.createTextNode((e instanceof Array) ? e[1] : e));
777
+ selectTag.appendChild(optionTag);
778
+ }.bind(this));
779
+ this.cached_selectTag = selectTag;
780
+ }
781
+
782
+ this.editField = this.cached_selectTag;
783
+ if(this.options.loadTextURL) this.loadExternalText();
784
+ this.form.appendChild(this.editField);
785
+ this.options.callback = function(form, value) {
786
+ return "value=" + encodeURIComponent(value);
787
+ }
788
+ }
789
+ });
790
+
791
+ // Delayed observer, like Form.Element.Observer,
792
+ // but waits for delay after last key input
793
+ // Ideal for live-search fields
794
+
795
+ Form.Element.DelayedObserver = Class.create();
796
+ Form.Element.DelayedObserver.prototype = {
797
+ initialize: function(element, delay, callback) {
798
+ this.delay = delay || 0.5;
799
+ this.element = $(element);
800
+ this.callback = callback;
801
+ this.timer = null;
802
+ this.lastValue = $F(this.element);
803
+ Event.observe(this.element,'keyup',this.delayedListener.bindAsEventListener(this));
804
+ },
805
+ delayedListener: function(event) {
806
+ if(this.lastValue == $F(this.element)) return;
807
+ if(this.timer) clearTimeout(this.timer);
808
+ this.timer = setTimeout(this.onTimerEvent.bind(this), this.delay * 1000);
809
+ this.lastValue = $F(this.element);
810
+ },
811
+ onTimerEvent: function() {
812
+ this.timer = null;
813
+ this.callback(this.element, $F(this.element));
814
+ }
815
+ };