edifice-forms 0.4.0 → 0.5.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (58) hide show
  1. data/README.md +22 -3
  2. data/Rakefile +6 -0
  3. data/app/assets/javascripts/edifice-forms/jquery-escape.js +15 -0
  4. data/app/assets/javascripts/edifice-forms/rails_form.js +1 -1
  5. data/app/assets/javascripts/edifice-forms/show-errors.js +9 -1
  6. data/edifice-forms.gemspec +7 -1
  7. data/lib/edifice-forms/responder.rb +4 -3
  8. data/lib/edifice-forms/version.rb +1 -1
  9. data/spec/error_handling_spec.rb +54 -0
  10. data/spec/form_model_spec.rb +25 -0
  11. data/spec/rails3.1/app/assets/images/rails.png +0 -0
  12. data/spec/rails3.1/app/assets/javascripts/application.js +10 -0
  13. data/spec/rails3.1/app/assets/javascripts/test/base.js +11 -0
  14. data/spec/rails3.1/app/assets/javascripts/vendor/jquery-pjax.js +264 -0
  15. data/spec/rails3.1/app/assets/stylesheets/application.css +7 -0
  16. data/spec/rails3.1/app/controllers/application_controller.rb +3 -0
  17. data/spec/rails3.1/app/controllers/users_controller.rb +18 -0
  18. data/spec/rails3.1/app/models/user.rb +15 -0
  19. data/spec/rails3.1/app/views/layouts/application.html.erb +14 -0
  20. data/spec/rails3.1/app/views/users/new.html.erb +15 -0
  21. data/spec/rails3.1/config/application.rb +55 -0
  22. data/spec/rails3.1/config/boot.rb +6 -0
  23. data/spec/rails3.1/config/environment.rb +5 -0
  24. data/spec/rails3.1/config/environments/test.rb +34 -0
  25. data/spec/rails3.1/config/initializers/backtrace_silencers.rb +7 -0
  26. data/spec/rails3.1/config/initializers/inflections.rb +10 -0
  27. data/spec/rails3.1/config/initializers/mime_types.rb +5 -0
  28. data/spec/rails3.1/config/initializers/secret_token.rb +7 -0
  29. data/spec/rails3.1/config/initializers/session_store.rb +8 -0
  30. data/spec/rails3.1/config/initializers/wrap_parameters.rb +14 -0
  31. data/spec/rails3.1/config/routes.rb +7 -0
  32. data/spec/rails3.1/log/.gitkeep +0 -0
  33. data/spec/rails_form_spec.rb +63 -0
  34. data/spec/spec_helper.rb +27 -0
  35. data/tmp/cache/assets/C93/520/sprockets%2Fc677444919ef374b958536e6f24b15b1 +0 -0
  36. data/tmp/cache/assets/CA5/380/sprockets%2F1ff55d945a664b750021ba61a96c0274 +0 -0
  37. data/tmp/cache/assets/CB2/300/sprockets%2Fc3870b357b9ca1b89950483165c0d8e0 +0 -0
  38. data/tmp/cache/assets/CD8/370/sprockets%2F357970feca3ac29060c1e3861e2c0953 +0 -0
  39. data/tmp/cache/assets/CDE/4B0/sprockets%2Fc589c30dd77520ba7fd4e16466a32192 +0 -0
  40. data/tmp/cache/assets/CEE/5F0/sprockets%2F54d38a9c813681d2838b96ebe140e84f +0 -0
  41. data/tmp/cache/assets/D19/8A0/sprockets%2F8870aaaed853f1b2d489b199327f802e +0 -0
  42. data/tmp/cache/assets/D24/3C0/sprockets%2F4b5f582b8f52d50f96d396837b44ace8 +0 -0
  43. data/tmp/cache/assets/D32/A10/sprockets%2F13fe41fee1fe35b49d145bcc06610705 +0 -0
  44. data/tmp/cache/assets/D4E/1B0/sprockets%2Ff7cbd26ba1d28d48de824f0e94586655 +0 -0
  45. data/tmp/cache/assets/D4F/C40/sprockets%2F6eef3882347ce72578caca5865b80cb6 +0 -0
  46. data/tmp/cache/assets/D50/850/sprockets%2F4acc8287556eb57079e6978b4a11bedf +0 -0
  47. data/tmp/cache/assets/D5A/EA0/sprockets%2Fd771ace226fc8215a3572e0aa35bb0d6 +0 -0
  48. data/tmp/cache/assets/D6F/BC0/sprockets%2Fcf7528da181e32b8e758e3d603cbba29 +0 -0
  49. data/tmp/cache/assets/D91/080/sprockets%2F3cc81f1bd8d23d9c1aab48c520e450f6 +0 -0
  50. data/tmp/cache/assets/DA6/150/sprockets%2F1eeb0387fb8841ef86accfe15932c72c +0 -0
  51. data/tmp/cache/assets/DAA/7E0/sprockets%2F423e3888dde6a807610fcaeb6de68c5e +0 -0
  52. data/tmp/cache/assets/DDC/400/sprockets%2Fcffd775d018f68ce5dba1ee0d951a994 +0 -0
  53. data/tmp/cache/assets/E04/890/sprockets%2F2f5173deea6c795b8fdde723bb4b63af +0 -0
  54. data/tmp/cache/assets/E26/AB0/sprockets%2Fc9abd9afeb3c158ad7aba136830f5cc6 +0 -0
  55. data/tmp/cache/assets/E2F/490/sprockets%2Fbee5b646eaa5f1f0ce53f413cf9bc59d +0 -0
  56. data/tmp/cache/assets/E66/560/sprockets%2Fd5e10c4f3e90a44eb2dffccfefe8594e +0 -0
  57. data/webrat.log +201 -0
  58. metadata +139 -8
data/README.md CHANGED
@@ -1,10 +1,27 @@
1
1
  Unobtrusive Javascript Form Extensions for Rails 3
2
2
  ==================================================
3
3
 
4
- edifice-forms the part of the [edifice project](https://github.com/tmeasday/edifice) which improves your experience with forms inside rails.
4
+ Edifice-widgets is a companion gem to [edifice](/tmeasday/edifice) which improves and simplifies your experience writing forms for rails.
5
5
 
6
6
  Note that it does not depend on edifice, although it complements it well.
7
7
 
8
+ Installation
9
+ ------------
10
+
11
+ To install, simply add to your Gemfile:
12
+
13
+ ```ruby
14
+ gem 'edifice-forms'
15
+ ```
16
+
17
+ To include the javascript, add to your application.js:
18
+
19
+ ```js
20
+ /*
21
+ *= require edifice-forms
22
+ */
23
+ ```
24
+
8
25
  Extending remote forms to handle errors
9
26
  ---------------------------------------
10
27
 
@@ -105,9 +122,11 @@ class Feedback < Edifice::Forms::FormModel
105
122
  validates :email, :presence => true, :format => {:with => /^.+@.+\..+$/}
106
123
  validates :message, :presence => true
107
124
 
125
+ after_save :deliver_feedback
126
+
108
127
  # if validations pass and we successfully save, go ahead and deliver the
109
128
  # feedback email to us, so we can read it.
110
- def save
129
+ def deliver_feedback
111
130
  SelfMailer.feedback(self).deliver
112
131
  end
113
132
  end
@@ -143,4 +162,4 @@ Simple, huh?
143
162
  License
144
163
  -------
145
164
 
146
- Edifice is crafted by [Percolate Studio](http://percolatestudio.com) and released under the [MIT license](www.opensource.org/licenses/MIT)
165
+ [Edifice](http://edifice-rails.com) is crafted by [Percolate Studio](http://percolatestudio.com) and released under the [MIT license](www.opensource.org/licenses/MIT)
data/Rakefile CHANGED
@@ -1,3 +1,9 @@
1
1
  require 'rubygems'
2
2
  require 'bundler'
3
3
  Bundler::GemHelper.install_tasks
4
+
5
+ require 'rspec/core/rake_task'
6
+ RSpec::Core::RakeTask.new(:spec)
7
+
8
+ task :test => :spec
9
+ task :default => :spec
@@ -0,0 +1,15 @@
1
+ // jquery.escape 1.0 - escape strings for use in jQuery selectors
2
+ // http://ianloic.com/tag/jquery.escape
3
+ // Copyright 2009 Ian McKellar <http://ian.mckellar.org/>
4
+ // Just like jQuery you can use it under either the MIT license or the GPL
5
+ // (see: http://docs.jquery.com/License)
6
+ (function() {
7
+ escape_re = /[#;&,\.\+\*~':"!\^\$\[\]\(\)=>|\/\\]/;
8
+ jQuery.escape = function jQuery$escape(s) {
9
+ var left = s.split(escape_re, 1)[0];
10
+ if (left == s) return s;
11
+ return left + '\\' +
12
+ s.substr(left.length, 1) +
13
+ jQuery.escape(s.substr(left.length+1));
14
+ }
15
+ })();
@@ -16,7 +16,7 @@ $.fn.rails_form = function(method) {
16
16
  // in these 'this' is the $form
17
17
  var methods = {
18
18
  fields: function() {
19
- return this.find('input, textarea, select');
19
+ return this.find('input:not([type=hidden]):not([type=submit]), textarea, select');
20
20
  },
21
21
 
22
22
  field: function(name_or_field) {
@@ -14,7 +14,15 @@
14
14
 
15
15
  } else if (/json/.test(contentType)) {
16
16
  // we will be receiving an error object back, we can pass it straight into rails_form.js
17
- $(this).rails_form('set_errors', $.parseJSON(request.responseText));
17
+ $('body').append('<pre>' + request.responseText + '</pre>');
18
+ var errors = $.parseJSON(request.responseText);
19
+
20
+ // they are using namespaced JSON
21
+ if ('errors' in errors && !$.isArray(errors['errors'])) {
22
+ errors = errors['errors']
23
+ }
24
+
25
+ $(this).rails_form('set_errors', errors);
18
26
  } else {
19
27
  throw "edifice-forms/show-errors: Don't know how to handle dataType " + request.dataType;
20
28
  }
@@ -7,7 +7,8 @@ Gem::Specification.new do |s|
7
7
  s.platform = Gem::Platform::RUBY
8
8
  s.author = 'Tom Coleman'
9
9
  s.email = 'tom@percolatestudio.com'
10
- s.summary = 'Unobtrusive JS Form extensions'
10
+ s.summary = 'Unobtrusive Rails JS Form extensions'
11
+ s.description = 'Edifice-widgets is a companion gem to edifice which improves and simplifies your experience writing forms for rails.'
11
12
 
12
13
  s.add_dependency 'jquery-rails'
13
14
 
@@ -15,4 +16,9 @@ Gem::Specification.new do |s|
15
16
  s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
16
17
  s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
17
18
  s.require_paths = ["lib"]
19
+
20
+ s.add_development_dependency 'rake'
21
+ s.add_development_dependency 'rails'
22
+ s.add_development_dependency 'rspec-rails'
23
+ s.add_development_dependency 'capybara'
18
24
  end
@@ -4,14 +4,15 @@ module Edifice
4
4
  class Responder < ActionController::Responder
5
5
  protected
6
6
  # add the :u_e header to xhr error requests
7
- def to_html
7
+ def navigation_behavior(error)
8
+ ::Rails.logger.warn '>>>>>>> has_errors? is: ' + (has_errors? ? 'true' : 'false')
8
9
  if controller.request.xhr? && !get? and has_errors? && default_action
9
10
  render :action => default_action, :status => :unprocessable_entity, :layout => nil
10
11
  else
11
- super
12
+ super(error)
12
13
  end
13
14
  end
14
-
15
+
15
16
  # actually render something on successful updates
16
17
  def to_format
17
18
  unless get? or has_errors? or post?
@@ -1,5 +1,5 @@
1
1
  module Edifice
2
2
  module Forms
3
- VERSION = "0.4.0"
3
+ VERSION = "0.5.0"
4
4
  end
5
5
  end
@@ -0,0 +1,54 @@
1
+ require File.dirname(__FILE__) + '/spec_helper'
2
+
3
+ shared_examples_for :form_that_shows_errors do
4
+ it "should show errors when validation fails" do
5
+ page.click_button('Submit')
6
+
7
+ page.should have_error_on 'user_name'
8
+ end
9
+
10
+ it "should show errors when validation fails in .errors" do
11
+ page.click_button('Submit')
12
+
13
+ page.should have_rendered_error_on 'user_name'
14
+ end
15
+
16
+ it "not change when validation succeeds" do
17
+ page.fill_in 'Name', :with => 'whatever'
18
+ page.click_button('Submit')
19
+
20
+ page.should have_no_error_on 'user_name'
21
+ end
22
+ end
23
+
24
+ describe 'html forms with show_errors', :type => :request, :js => true do
25
+ before(:each) do
26
+ page.visit('/users/new?type=html&ajax=false')
27
+ end
28
+
29
+ it_behaves_like :form_that_shows_errors
30
+ end
31
+
32
+ describe 'json forms with show_errors', :type => :request, :js => true do
33
+ before(:each) do
34
+ page.visit('/users/new?type=json&ajax=false')
35
+ end
36
+
37
+ it_behaves_like :form_that_shows_errors
38
+ end
39
+
40
+ describe 'ajax html forms with show_errors', :type => :request, :js => true do
41
+ before(:each) do
42
+ page.visit('/users/new?type=html&ajax=true')
43
+ end
44
+
45
+ it_behaves_like :form_that_shows_errors
46
+ end
47
+
48
+ describe 'ajax json forms with show_errors', :type => :request, :js => true do
49
+ before(:each) do
50
+ page.visit('/users/new?type=json&ajax=true')
51
+ end
52
+
53
+ it_behaves_like :form_that_shows_errors
54
+ end
@@ -0,0 +1,25 @@
1
+ require File.dirname(__FILE__) + '/spec_helper'
2
+
3
+ describe UsersController do
4
+ include RSpec::Rails::ControllerExampleGroup
5
+
6
+ it 'should return new' do
7
+ get 'new'
8
+
9
+ response.should render_template(:new)
10
+ end
11
+
12
+ it 'should behave correctly on validation errors' do
13
+ post 'create', :user => {:name => ''}
14
+
15
+ response.should render_template(:new)
16
+ end
17
+
18
+ it 'should update successfully' do
19
+ User.should_receive(:model_saved)
20
+
21
+ post 'create', :user => {:name => 'a lengthy name'}
22
+
23
+ response.should redirect_to(users_path)
24
+ end
25
+ end
@@ -0,0 +1,10 @@
1
+ // This is a manifest file that'll be compiled into including all the files listed below.
2
+ // Add new JavaScript/Coffee code in separate files in this directory and they'll automatically
3
+ // be included in the compiled file accessible from http://example.com/assets/application.js
4
+ // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the
5
+ // the compiled file.
6
+ //
7
+ //= require jquery
8
+ //= require jquery_ujs
9
+ //= require edifice-forms
10
+ //= require_tree .
@@ -0,0 +1,11 @@
1
+ (function($) {
2
+ window.testBase = {
3
+ onReady: function() {
4
+ $('body').append('<h1 class="ready">');
5
+ },
6
+
7
+ onLoad: function() {
8
+ $('body').append('<h1 class="load">');
9
+ }
10
+ }
11
+ }(jQuery));
@@ -0,0 +1,264 @@
1
+ // jquery.pjax.js
2
+ // copyright chris wanstrath
3
+ // https://github.com/defunkt/jquery-pjax
4
+
5
+ (function($){
6
+
7
+ // When called on a link, fetches the href with ajax into the
8
+ // container specified as the first parameter or with the data-pjax
9
+ // attribute on the link itself.
10
+ //
11
+ // Tries to make sure the back button and ctrl+click work the way
12
+ // you'd expect.
13
+ //
14
+ // Accepts a jQuery ajax options object that may include these
15
+ // pjax specific options:
16
+ //
17
+ // container - Where to stick the response body. Usually a String selector.
18
+ // $(container).html(xhr.responseBody)
19
+ // push - Whether to pushState the URL. Defaults to true (of course).
20
+ // replace - Want to use replaceState instead? That's cool.
21
+ //
22
+ // For convenience the first parameter can be either the container or
23
+ // the options object.
24
+ //
25
+ // Returns the jQuery object
26
+ $.fn.pjax = function( container, options ) {
27
+ if ( options )
28
+ options.container = container
29
+ else
30
+ options = $.isPlainObject(container) ? container : {container:container}
31
+
32
+ // We can't persist $objects using the history API so we must use
33
+ // a String selector. Bail if we got anything else.
34
+ if ( options.container && typeof options.container !== 'string' ) {
35
+ throw "pjax container must be a string selector!"
36
+ return false
37
+ }
38
+
39
+ return this.live('click', function(event){
40
+ // Middle click, cmd click, and ctrl click should open
41
+ // links in a new tab as normal.
42
+ if ( event.which > 1 || event.metaKey )
43
+ return true
44
+
45
+ var defaults = {
46
+ url: this.href,
47
+ container: $(this).attr('data-pjax'),
48
+ clickedElement: $(this),
49
+ fragment: null
50
+ }
51
+
52
+ $.pjax($.extend({}, defaults, options))
53
+
54
+ event.preventDefault()
55
+ })
56
+ }
57
+
58
+
59
+ // Loads a URL with ajax, puts the response body inside a container,
60
+ // then pushState()'s the loaded URL.
61
+ //
62
+ // Works just like $.ajax in that it accepts a jQuery ajax
63
+ // settings object (with keys like url, type, data, etc).
64
+ //
65
+ // Accepts these extra keys:
66
+ //
67
+ // container - Where to stick the response body. Must be a String.
68
+ // $(container).html(xhr.responseBody)
69
+ // push - Whether to pushState the URL. Defaults to true (of course).
70
+ // replace - Want to use replaceState instead? That's cool.
71
+ //
72
+ // Use it just like $.ajax:
73
+ //
74
+ // var xhr = $.pjax({ url: this.href, container: '#main' })
75
+ // console.log( xhr.readyState )
76
+ //
77
+ // Returns whatever $.ajax returns.
78
+ var pjax = $.pjax = function( options ) {
79
+ var $container = $(options.container),
80
+ success = options.success || $.noop
81
+
82
+ // We don't want to let anyone override our success handler.
83
+ delete options.success
84
+
85
+ // We can't persist $objects using the history API so we must use
86
+ // a String selector. Bail if we got anything else.
87
+ if ( typeof options.container !== 'string' )
88
+ throw "pjax container must be a string selector!"
89
+
90
+ options = $.extend(true, {}, pjax.defaults, options)
91
+
92
+ if ( $.isFunction(options.url) ) {
93
+ options.url = options.url()
94
+ }
95
+
96
+ options.context = $container
97
+
98
+ options.success = function(data){
99
+ if ( options.fragment ) {
100
+ // If they specified a fragment, look for it in the response
101
+ // and pull it out.
102
+ var $fragment = $(data).find(options.fragment)
103
+ if ( $fragment.length )
104
+ data = $fragment.children()
105
+ else
106
+ return window.location = options.url
107
+ } else {
108
+ // If we got no data or an entire web page, go directly
109
+ // to the page and let normal error handling happen.
110
+ if ( !$.trim(data) || /<html/i.test(data) )
111
+ return window.location = options.url
112
+ }
113
+
114
+ // Make it happen.
115
+ this.html(data)
116
+
117
+ // If there's a <title> tag in the response, use it as
118
+ // the page's title.
119
+ var oldTitle = document.title,
120
+ title = $.trim( this.find('title').remove().text() )
121
+ if ( title ) document.title = title
122
+
123
+ // No <title>? Fragment? Look for data-title and title attributes.
124
+ if ( !title && options.fragment ) {
125
+ title = $fragment.attr('title') || $fragment.data('title')
126
+ }
127
+
128
+ var state = {
129
+ pjax: options.container,
130
+ fragment: options.fragment,
131
+ timeout: options.timeout
132
+ }
133
+
134
+ // If there are extra params, save the complete URL in the state object
135
+ var query = $.param(options.data)
136
+ if ( query != "_pjax=true" )
137
+ state.url = options.url + (/\?/.test(options.url) ? "&" : "?") + query
138
+
139
+ if ( options.replace ) {
140
+ window.history.replaceState(state, document.title, options.url)
141
+ } else if ( options.push ) {
142
+ // this extra replaceState before first push ensures good back
143
+ // button behavior
144
+ if ( !pjax.active ) {
145
+ window.history.replaceState($.extend({}, state, {url:null}), oldTitle)
146
+ pjax.active = true
147
+ }
148
+
149
+ window.history.pushState(state, document.title, options.url)
150
+ }
151
+
152
+ // Google Analytics support
153
+ if ( (options.replace || options.push) && window._gaq )
154
+ _gaq.push(['_trackPageview'])
155
+
156
+ // If the URL has a hash in it, make sure the browser
157
+ // knows to navigate to the hash.
158
+ var hash = window.location.hash.toString()
159
+ if ( hash !== '' ) {
160
+ window.location.href = hash
161
+ }
162
+
163
+ // Invoke their success handler if they gave us one.
164
+ success.apply(this, arguments)
165
+ }
166
+
167
+ // Cancel the current request if we're already pjaxing
168
+ var xhr = pjax.xhr
169
+ if ( xhr && xhr.readyState < 4) {
170
+ xhr.onreadystatechange = $.noop
171
+ xhr.abort()
172
+ }
173
+
174
+ pjax.options = options
175
+ pjax.xhr = $.ajax(options)
176
+ $(document).trigger('pjax', [pjax.xhr, options])
177
+
178
+ return pjax.xhr
179
+ }
180
+
181
+
182
+ pjax.defaults = {
183
+ timeout: 650,
184
+ push: true,
185
+ replace: false,
186
+ // We want the browser to maintain two separate internal caches: one for
187
+ // pjax'd partial page loads and one for normal page loads. Without
188
+ // adding this secret parameter, some browsers will often confuse the two.
189
+ data: { _pjax: true },
190
+ type: 'GET',
191
+ dataType: 'html',
192
+ beforeSend: function(xhr){
193
+ this.trigger('pjax:start', [xhr, pjax.options])
194
+ // start.pjax is deprecated
195
+ this.trigger('start.pjax', [xhr, pjax.options])
196
+ xhr.setRequestHeader('X-PJAX', 'true')
197
+ },
198
+ error: function(xhr, textStatus, errorThrown){
199
+ if ( textStatus !== 'abort' )
200
+ window.location = pjax.options.url
201
+ },
202
+ complete: function(xhr){
203
+ this.trigger('pjax:end', [xhr, pjax.options])
204
+ // end.pjax is deprecated
205
+ this.trigger('end.pjax', [xhr, pjax.options])
206
+ }
207
+ }
208
+
209
+
210
+ // Used to detect initial (useless) popstate.
211
+ // If history.state exists, assume browser isn't going to fire initial popstate.
212
+ var popped = ('state' in window.history), initialURL = location.href
213
+
214
+
215
+ // popstate handler takes care of the back and forward buttons
216
+ //
217
+ // You probably shouldn't use pjax on pages with other pushState
218
+ // stuff yet.
219
+ $(window).bind('popstate', function(event){
220
+ // Ignore inital popstate that some browsers fire on page load
221
+ var initialPop = !popped && location.href == initialURL
222
+ popped = true
223
+ if ( initialPop ) return
224
+
225
+ var state = event.state
226
+
227
+ if ( state && state.pjax ) {
228
+ var container = state.pjax
229
+ if ( $(container+'').length )
230
+ $.pjax({
231
+ url: state.url || location.href,
232
+ fragment: state.fragment,
233
+ container: container,
234
+ push: false,
235
+ timeout: state.timeout
236
+ })
237
+ else
238
+ window.location = location.href
239
+ }
240
+ })
241
+
242
+
243
+ // Add the state property to jQuery's event object so we can use it in
244
+ // $(window).bind('popstate')
245
+ if ( $.inArray('state', $.event.props) < 0 )
246
+ $.event.props.push('state')
247
+
248
+
249
+ // Is pjax supported by this browser?
250
+ $.support.pjax =
251
+ window.history && window.history.pushState && window.history.replaceState
252
+ // pushState isn't reliable on iOS yet.
253
+ && !navigator.userAgent.match(/(iPod|iPhone|iPad|WebApps\/.+CFNetwork)/)
254
+
255
+
256
+ // Fall back to normalcy for older browsers.
257
+ if ( !$.support.pjax ) {
258
+ $.pjax = function( options ) {
259
+ window.location = $.isFunction(options.url) ? options.url() : options.url
260
+ }
261
+ $.fn.pjax = function() { return this }
262
+ }
263
+
264
+ })(jQuery);
@@ -0,0 +1,7 @@
1
+ /*
2
+ * This is a manifest file that'll automatically include all the stylesheets available in this directory
3
+ * and any sub-directories. You're free to add application-wide styles to this file and they'll appear at
4
+ * the top of the compiled file, but it's generally better to create a new file per style scope.
5
+ *= require_self
6
+ *= require_tree .
7
+ */
@@ -0,0 +1,3 @@
1
+ class ApplicationController < ActionController::Base
2
+ protect_from_forgery
3
+ end
@@ -0,0 +1,18 @@
1
+ class UsersController < ApplicationController
2
+ respond_to :html, :json
3
+
4
+ def new
5
+ @type = params[:type] || 'html'
6
+ @ajax = params[:ajax] || false
7
+
8
+ respond_with @user = User.new
9
+ end
10
+
11
+ def create
12
+ respond_with @user = User.create(params[:user])
13
+ end
14
+
15
+ def index
16
+ head :ok
17
+ end
18
+ end
@@ -0,0 +1,15 @@
1
+ class User < Edifice::Forms::FormModel
2
+ attr_accessor :name, :location, :description
3
+
4
+ validates :name, :length => {:minimum => 3}, :presence => true
5
+
6
+ after_save :log_save
7
+
8
+ def log_save
9
+ User.model_saved
10
+ end
11
+
12
+ # this is just here so we can easily test for it.
13
+ def self.model_saved
14
+ end
15
+ end
@@ -0,0 +1,14 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Rails31</title>
5
+ <%= stylesheet_link_tag "application" %>
6
+ <%= javascript_include_tag "application" %>
7
+ <%= csrf_meta_tags %>
8
+ </head>
9
+ <body>
10
+
11
+ <%= yield %>
12
+
13
+ </body>
14
+ </html>
@@ -0,0 +1,15 @@
1
+ <%= form_for @user, :remote => @ajax,
2
+ :html => {:'data-form' => 'show_errors', :'data-type' => @type} do |f| %>
3
+ <%= render_errors(f) %>
4
+
5
+ <%= f.label :name %>
6
+ <%= f.text_field :name %>
7
+
8
+ <%= f.label :location %>
9
+ <%= f.text_field :location %>
10
+
11
+ <%= f.label :description %>
12
+ <%= f.text_field :description %>
13
+
14
+ <button type="submit">Submit</button>
15
+ <% end %>
@@ -0,0 +1,55 @@
1
+ require File.expand_path('../boot', __FILE__)
2
+
3
+ require "action_controller/railtie"
4
+ require "sprockets/railtie"
5
+
6
+ require 'jquery-rails'
7
+ require 'edifice-forms'
8
+
9
+ if defined?(Bundler)
10
+ # If you precompile assets before deploying to production, use this line
11
+ Bundler.require(*Rails.groups(:assets => %w(development test)))
12
+ # If you want your assets lazily compiled in production, use this line
13
+ # Bundler.require(:default, :assets, Rails.env)
14
+ end
15
+
16
+ module Rails31
17
+ class Application < Rails::Application
18
+ # don't know how i was supposed to set this
19
+ config.root = ENV['RAILS_ROOT']
20
+
21
+ # Settings in config/environments/* take precedence over those specified here.
22
+ # Application configuration should go into files in config/initializers
23
+ # -- all .rb files in that directory are automatically loaded.
24
+
25
+ # Custom directories with classes and modules you want to be autoloadable.
26
+ # config.autoload_paths += %W(#{config.root}/extras)
27
+
28
+ # Only load the plugins named here, in the order given (default is alphabetical).
29
+ # :all can be used as a placeholder for all plugins not explicitly named.
30
+ # config.plugins = [ :exception_notification, :ssl_requirement, :all ]
31
+
32
+ # Activate observers that should always be running.
33
+ # config.active_record.observers = :cacher, :garbage_collector, :forum_observer
34
+
35
+ # Set Time.zone default to the specified zone and make Active Record auto-convert to this zone.
36
+ # Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC.
37
+ # config.time_zone = 'Central Time (US & Canada)'
38
+
39
+ # The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded.
40
+ # config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s]
41
+ # config.i18n.default_locale = :de
42
+
43
+ # Configure the default encoding used in templates for Ruby 1.9.
44
+ config.encoding = "utf-8"
45
+
46
+ # Configure sensitive parameters which will be filtered from the log file.
47
+ config.filter_parameters += [:password]
48
+
49
+ # Enable the asset pipeline
50
+ config.assets.enabled = true
51
+
52
+ # Version of your assets, change this if you want to expire all your assets
53
+ config.assets.version = '1.0'
54
+ end
55
+ end
@@ -0,0 +1,6 @@
1
+ require 'rubygems'
2
+
3
+ # Set up gems listed in the Gemfile.
4
+ ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__)
5
+
6
+ require 'bundler/setup' if File.exists?(ENV['BUNDLE_GEMFILE'])
@@ -0,0 +1,5 @@
1
+ # Load the rails application
2
+ require File.expand_path('../application', __FILE__)
3
+
4
+ # Initialize the rails application
5
+ Rails31::Application.initialize!