simple_form 4.0.0 → 5.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/CHANGELOG.md +41 -0
- data/MIT-LICENSE +1 -1
- data/README.md +12 -13
- data/lib/generators/simple_form/templates/config/initializers/simple_form.rb +0 -6
- data/lib/generators/simple_form/templates/config/initializers/simple_form_bootstrap.rb +24 -23
- data/lib/simple_form/components/errors.rb +5 -1
- data/lib/simple_form/form_builder.rb +29 -10
- data/lib/simple_form/inputs/base.rb +4 -1
- data/lib/simple_form/inputs/boolean_input.rb +1 -0
- data/lib/simple_form/inputs/color_input.rb +14 -0
- data/lib/simple_form/inputs/priority_input.rb +0 -4
- data/lib/simple_form/inputs/string_input.rb +1 -1
- data/lib/simple_form/inputs.rb +1 -0
- data/lib/simple_form/tags.rb +6 -2
- data/lib/simple_form/version.rb +1 -1
- data/lib/simple_form.rb +21 -4
- data/test/action_view_extensions/builder_test.rb +22 -4
- data/test/form_builder/association_test.rb +6 -0
- data/test/form_builder/error_test.rb +6 -0
- data/test/form_builder/general_test.rb +19 -20
- data/test/form_builder/input_field_test.rb +3 -9
- data/test/form_builder/wrapper_test.rb +1 -1
- data/test/inputs/boolean_input_test.rb +8 -0
- data/test/inputs/collection_check_boxes_input_test.rb +8 -0
- data/test/inputs/collection_radio_buttons_input_test.rb +8 -0
- data/test/inputs/color_input_test.rb +10 -0
- data/test/inputs/datetime_input_test.rb +2 -12
- data/test/inputs/disabled_test.rb +13 -0
- data/test/inputs/discovery_test.rb +21 -0
- data/test/inputs/priority_input_test.rb +6 -14
- data/test/inputs/string_input_test.rb +8 -15
- data/test/support/discovery_inputs.rb +7 -0
- data/test/support/misc_helpers.rb +6 -0
- data/test/support/models.rb +25 -6
- data/test/test_helper.rb +6 -3
- metadata +40 -37
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
|
-
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: a1bfd4d5f14c750caf9e2fa6287033c0331473215737a96b4bcece8e6f84a7a9
|
4
|
+
data.tar.gz: 9c09b014e449f5b8823c062092b864e538c638fd418d782919a9c0a3cd50df78
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 620056c8b34f6f93229b2ee0771adb0aa7d267fcc22b9d94c214f62287061930e9b0e35a648f47a890452cb8098bc57643a1dce7bbed3df3c2058546b58c5090
|
7
|
+
data.tar.gz: 4fa4e5a106b9065405fbba77b8a84ed99e86d84ea4a4435d0a459ee0196a020afbec0ec62fcd2c79f8885d771e48700409ae055ea4858d15b2b5ddd9c3fb5168
|
data/CHANGELOG.md
CHANGED
@@ -1,5 +1,46 @@
|
|
1
1
|
## Unreleased
|
2
2
|
|
3
|
+
## 5.0.0
|
4
|
+
|
5
|
+
### Enhancements
|
6
|
+
* Set multiple attribute for grouped selects also. [@ollym](https://github.com/ollym)
|
7
|
+
* Removes or renames label classes. [Abduvakilov](https://github.com/Abduvakilov)
|
8
|
+
* Support to label custom classes for inline collections. [@feliperenan](https://github.com/feliperenan)
|
9
|
+
* Update bootstrap generator template to match v4.3.x. [@m5o](https://github.com/m5o)
|
10
|
+
* Allow "required" attribute in generated select elements of PriorityInput. [@mcountis](https://github.com/mcountis)
|
11
|
+
|
12
|
+
### Bug fix
|
13
|
+
* Do not call `#send` in form object to check whether the attribute is a file input. [@tegon](https://github.com/tegon)
|
14
|
+
|
15
|
+
## Deprecations
|
16
|
+
* The config `SimpleForm.file_methods` is deprecated and it has no effect. Simple Form now supports automatically discover of file inputs for the following Gems: activestorage, carrierwave, paperclip, refile and shrine. If you are using a custom method that is not from one of the supported Gems, please change your forms to pass the input type explicitly:
|
17
|
+
|
18
|
+
```erb
|
19
|
+
<%= form.input :avatar, as: :file %>
|
20
|
+
```
|
21
|
+
|
22
|
+
See http://blog.plataformatec.com.br/2019/09/incorrect-access-control-in-simple-form-cve-2019-16676 for more information.
|
23
|
+
|
24
|
+
## 4.1.0
|
25
|
+
|
26
|
+
### Enhancements
|
27
|
+
* Guess input type more carefully. [@sringling](https://github.com/sringling)
|
28
|
+
* Allow custom error on forms without model. [@victorperez](https://github.com/victorperez)
|
29
|
+
* Do not support Ruby < 2.3 anymore. [@gssbzn](https://github.com/gssbzn)
|
30
|
+
* Add color input type. [@gssbzn](https://github.com/gssbzn)
|
31
|
+
|
32
|
+
### Bug fix
|
33
|
+
* Improve disabled option to input_field. [@betelgeuse](https://github.com/betelgeuse)
|
34
|
+
* Memoize `input_html_classes` in `SimpleForm::Inputs::Base`. [@RigoTheDev](https://github.com/RigoTheDev)
|
35
|
+
* Fix column type citext HTML5 input type bug. [@brucew](https://github.com/brucew)
|
36
|
+
* Use form attribute in the nested boolean hidden field when it is given. [@feliperenan](https://github.com/feliperenan)
|
37
|
+
|
38
|
+
## 4.0.1
|
39
|
+
|
40
|
+
### Bug fix
|
41
|
+
* Do not support Rails 4 anymore. [@rafaelfranca](https://github.com/rafaelfranca)
|
42
|
+
* Add missing comma. [@vill](https://github.com/vill)
|
43
|
+
|
3
44
|
## 4.0.0
|
4
45
|
|
5
46
|
### Enhancements
|
data/MIT-LICENSE
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
Copyright (c) 2009-
|
1
|
+
Copyright (c) 2009-2019 Plataformatec http://plataformatec.com.br/
|
2
2
|
|
3
3
|
Permission is hereby granted, free of charge, to any person obtaining
|
4
4
|
a copy of this software and associated documentation files (the
|
data/README.md
CHANGED
@@ -288,13 +288,6 @@ end
|
|
288
288
|
</form>
|
289
289
|
```
|
290
290
|
|
291
|
-
Produces:
|
292
|
-
|
293
|
-
```html
|
294
|
-
<input class="string required" id="user_name" maxlength="100"
|
295
|
-
name="user[name]" size="100" type="text" value="Carlos" />
|
296
|
-
```
|
297
|
-
|
298
291
|
To view the actual RDocs for this, check them out here - http://rubydoc.info/github/plataformatec/simple_form/master/SimpleForm/FormBuilder:input_field
|
299
292
|
|
300
293
|
### Collections
|
@@ -507,7 +500,7 @@ Creates a collection of radio inputs with labels associated (same API as `collec
|
|
507
500
|
|
508
501
|
```ruby
|
509
502
|
form_for @user do |f|
|
510
|
-
f.collection_radio_buttons :options, [[true, 'Yes']
|
503
|
+
f.collection_radio_buttons :options, [[true, 'Yes'], [false, 'No']], :first, :last
|
511
504
|
end
|
512
505
|
```
|
513
506
|
|
@@ -524,7 +517,7 @@ Creates a collection of checkboxes with labels associated (same API as `collecti
|
|
524
517
|
|
525
518
|
```ruby
|
526
519
|
form_for @user do |f|
|
527
|
-
f.collection_check_boxes :options, [[true, 'Yes']
|
520
|
+
f.collection_check_boxes :options, [[true, 'Yes'], [false, 'No']], :first, :last
|
528
521
|
end
|
529
522
|
```
|
530
523
|
|
@@ -545,6 +538,12 @@ form_for @user do |f|
|
|
545
538
|
end
|
546
539
|
```
|
547
540
|
|
541
|
+
To add a CSS class to the label item, you can use the `item_label_class` option:
|
542
|
+
|
543
|
+
```ruby
|
544
|
+
f.collection_check_boxes :role_ids, Role.all, :id, :name, item_label_class: 'my-custom-class'
|
545
|
+
```
|
546
|
+
|
548
547
|
## Available input types and defaults for each column type
|
549
548
|
|
550
549
|
The following table shows the html element you will get for each attribute
|
@@ -910,8 +909,8 @@ config.wrappers do |b|
|
|
910
909
|
end
|
911
910
|
```
|
912
911
|
|
913
|
-
This
|
914
|
-
|
912
|
+
This sets the input and label classes to `'label-input-class'` and will set the class `'is-invalid'`
|
913
|
+
if the input has errors and `'is-valid'` if the input is valid.
|
915
914
|
|
916
915
|
If you want to customize the custom _Form components_ on demand you can give it a name like this:
|
917
916
|
|
@@ -1144,7 +1143,7 @@ by passing the html5 option:
|
|
1144
1143
|
### Using non Active Record objects
|
1145
1144
|
|
1146
1145
|
There are few ways to build forms with objects that don't inherit from Active Record, as
|
1147
|
-
|
1146
|
+
follows:
|
1148
1147
|
|
1149
1148
|
You can include the module `ActiveModel::Model`.
|
1150
1149
|
|
@@ -1248,7 +1247,7 @@ https://github.com/plataformatec/simple_form/issues
|
|
1248
1247
|
|
1249
1248
|
## License
|
1250
1249
|
|
1251
|
-
MIT License. Copyright 2009-
|
1250
|
+
MIT License. Copyright 2009-2019 Plataformatec. http://plataformatec.com.br
|
1252
1251
|
|
1253
1252
|
You are not granted rights or licenses to the trademarks of the Plataformatec, including without
|
1254
1253
|
limitation the Simple Form name or logo.
|
@@ -87,9 +87,6 @@ SimpleForm.setup do |config|
|
|
87
87
|
# CSS class to add for error notification helper.
|
88
88
|
config.error_notification_class = 'error_notification'
|
89
89
|
|
90
|
-
# ID to add for error notification helper.
|
91
|
-
# config.error_notification_id = nil
|
92
|
-
|
93
90
|
# Series of attempts to detect a default label method for collection.
|
94
91
|
# config.collection_label_methods = [ :to_label, :name, :title, :to_s ]
|
95
92
|
|
@@ -132,9 +129,6 @@ SimpleForm.setup do |config|
|
|
132
129
|
# change this configuration to true.
|
133
130
|
config.browser_validations = false
|
134
131
|
|
135
|
-
# Collection of methods to detect if a file type was given.
|
136
|
-
# config.file_methods = [ :mounted_as, :file?, :public_filename :attached? ]
|
137
|
-
|
138
132
|
# Custom mappings for input types. This should be a hash containing a regexp
|
139
133
|
# to match as key, and the input type that will be used when the field name
|
140
134
|
# matches the regexp as value.
|
@@ -57,7 +57,7 @@ SimpleForm.setup do |config|
|
|
57
57
|
b.optional :pattern
|
58
58
|
b.optional :min_max
|
59
59
|
b.optional :readonly
|
60
|
-
b.use :label
|
60
|
+
b.use :label
|
61
61
|
b.use :input, class: 'form-control', error_class: 'is-invalid', valid_class: 'is-valid'
|
62
62
|
b.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback' }
|
63
63
|
b.use :hint, wrap_with: { tag: 'small', class: 'form-text text-muted' }
|
@@ -76,7 +76,7 @@ SimpleForm.setup do |config|
|
|
76
76
|
end
|
77
77
|
|
78
78
|
# vertical input for radio buttons and check boxes
|
79
|
-
config.wrappers :vertical_collection, item_wrapper_class: 'form-check', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
79
|
+
config.wrappers :vertical_collection, item_wrapper_class: 'form-check', item_label_class: 'form-check-label', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
80
80
|
b.use :html5
|
81
81
|
b.optional :readonly
|
82
82
|
b.wrapper :legend_tag, tag: 'legend', class: 'col-form-label pt-0' do |ba|
|
@@ -88,7 +88,7 @@ SimpleForm.setup do |config|
|
|
88
88
|
end
|
89
89
|
|
90
90
|
# vertical input for inline radio buttons and check boxes
|
91
|
-
config.wrappers :vertical_collection_inline, item_wrapper_class: 'form-check form-check-inline', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
91
|
+
config.wrappers :vertical_collection_inline, item_wrapper_class: 'form-check form-check-inline', item_label_class: 'form-check-label', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
92
92
|
b.use :html5
|
93
93
|
b.optional :readonly
|
94
94
|
b.wrapper :legend_tag, tag: 'legend', class: 'col-form-label pt-0' do |ba|
|
@@ -108,7 +108,7 @@ SimpleForm.setup do |config|
|
|
108
108
|
b.optional :readonly
|
109
109
|
b.use :label
|
110
110
|
b.use :input, class: 'form-control-file', error_class: 'is-invalid', valid_class: 'is-valid'
|
111
|
-
b.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback
|
111
|
+
b.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback' }
|
112
112
|
b.use :hint, wrap_with: { tag: 'small', class: 'form-text text-muted' }
|
113
113
|
end
|
114
114
|
|
@@ -116,7 +116,7 @@ SimpleForm.setup do |config|
|
|
116
116
|
config.wrappers :vertical_multi_select, tag: 'div', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
117
117
|
b.use :html5
|
118
118
|
b.optional :readonly
|
119
|
-
b.use :label
|
119
|
+
b.use :label
|
120
120
|
b.wrapper tag: 'div', class: 'd-flex flex-row justify-content-between align-items-center' do |ba|
|
121
121
|
ba.use :input, class: 'form-control mx-1', error_class: 'is-invalid', valid_class: 'is-valid'
|
122
122
|
end
|
@@ -174,10 +174,10 @@ SimpleForm.setup do |config|
|
|
174
174
|
end
|
175
175
|
|
176
176
|
# horizontal input for radio buttons and check boxes
|
177
|
-
config.wrappers :horizontal_collection, item_wrapper_class: 'form-check', tag: 'div', class: 'form-group row', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
177
|
+
config.wrappers :horizontal_collection, item_wrapper_class: 'form-check', item_label_class: 'form-check-label', tag: 'div', class: 'form-group row', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
178
178
|
b.use :html5
|
179
179
|
b.optional :readonly
|
180
|
-
b.use :label, class: 'col-sm-3 form-
|
180
|
+
b.use :label, class: 'col-sm-3 col-form-label pt-0'
|
181
181
|
b.wrapper :grid_wrapper, tag: 'div', class: 'col-sm-9' do |ba|
|
182
182
|
ba.use :input, class: 'form-check-input', error_class: 'is-invalid', valid_class: 'is-valid'
|
183
183
|
ba.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback d-block' }
|
@@ -186,10 +186,10 @@ SimpleForm.setup do |config|
|
|
186
186
|
end
|
187
187
|
|
188
188
|
# horizontal input for inline radio buttons and check boxes
|
189
|
-
config.wrappers :horizontal_collection_inline, item_wrapper_class: 'form-check form-check-inline', tag: 'div', class: 'form-group row', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
189
|
+
config.wrappers :horizontal_collection_inline, item_wrapper_class: 'form-check form-check-inline', item_label_class: 'form-check-label', tag: 'div', class: 'form-group row', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
190
190
|
b.use :html5
|
191
191
|
b.optional :readonly
|
192
|
-
b.use :label, class: 'col-sm-3 form-
|
192
|
+
b.use :label, class: 'col-sm-3 col-form-label pt-0'
|
193
193
|
b.wrapper :grid_wrapper, tag: 'div', class: 'col-sm-9' do |ba|
|
194
194
|
ba.use :input, class: 'form-check-input', error_class: 'is-invalid', valid_class: 'is-valid'
|
195
195
|
ba.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback d-block' }
|
@@ -204,7 +204,7 @@ SimpleForm.setup do |config|
|
|
204
204
|
b.optional :maxlength
|
205
205
|
b.optional :minlength
|
206
206
|
b.optional :readonly
|
207
|
-
b.use :label, class: 'col-sm-3 form-
|
207
|
+
b.use :label, class: 'col-sm-3 col-form-label'
|
208
208
|
b.wrapper :grid_wrapper, tag: 'div', class: 'col-sm-9' do |ba|
|
209
209
|
ba.use :input, error_class: 'is-invalid', valid_class: 'is-valid'
|
210
210
|
ba.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback d-block' }
|
@@ -216,7 +216,7 @@ SimpleForm.setup do |config|
|
|
216
216
|
config.wrappers :horizontal_multi_select, tag: 'div', class: 'form-group row', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
217
217
|
b.use :html5
|
218
218
|
b.optional :readonly
|
219
|
-
b.use :label, class: 'col-sm-3
|
219
|
+
b.use :label, class: 'col-sm-3 col-form-label'
|
220
220
|
b.wrapper :grid_wrapper, tag: 'div', class: 'col-sm-9' do |ba|
|
221
221
|
ba.wrapper tag: 'div', class: 'd-flex flex-row justify-content-between align-items-center' do |bb|
|
222
222
|
bb.use :input, class: 'form-control mx-1', error_class: 'is-invalid', valid_class: 'is-valid'
|
@@ -232,7 +232,7 @@ SimpleForm.setup do |config|
|
|
232
232
|
b.use :placeholder
|
233
233
|
b.optional :readonly
|
234
234
|
b.optional :step
|
235
|
-
b.use :label, class: 'col-sm-3 form-
|
235
|
+
b.use :label, class: 'col-sm-3 col-form-label'
|
236
236
|
b.wrapper :grid_wrapper, tag: 'div', class: 'col-sm-9' do |ba|
|
237
237
|
ba.use :input, class: 'form-control-range', error_class: 'is-invalid', valid_class: 'is-valid'
|
238
238
|
ba.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback d-block' }
|
@@ -260,7 +260,7 @@ SimpleForm.setup do |config|
|
|
260
260
|
end
|
261
261
|
|
262
262
|
# inline input for boolean
|
263
|
-
config.wrappers :inline_boolean, tag: 'span', class: 'form-check
|
263
|
+
config.wrappers :inline_boolean, tag: 'span', class: 'form-check mb-2 mr-sm-2', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
264
264
|
b.use :html5
|
265
265
|
b.optional :readonly
|
266
266
|
b.use :input, class: 'form-check-input', error_class: 'is-invalid', valid_class: 'is-valid'
|
@@ -284,10 +284,11 @@ SimpleForm.setup do |config|
|
|
284
284
|
end
|
285
285
|
end
|
286
286
|
|
287
|
+
# custom input switch for boolean
|
287
288
|
config.wrappers :custom_boolean_switch, tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
288
289
|
b.use :html5
|
289
290
|
b.optional :readonly
|
290
|
-
b.wrapper :form_check_wrapper, tag: 'div', class: 'custom-control custom-
|
291
|
+
b.wrapper :form_check_wrapper, tag: 'div', class: 'custom-control custom-switch' do |bb|
|
291
292
|
bb.use :input, class: 'custom-control-input', error_class: 'is-invalid', valid_class: 'is-valid'
|
292
293
|
bb.use :label, class: 'custom-control-label'
|
293
294
|
bb.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback' }
|
@@ -296,7 +297,7 @@ SimpleForm.setup do |config|
|
|
296
297
|
end
|
297
298
|
|
298
299
|
# custom input for radio buttons and check boxes
|
299
|
-
config.wrappers :custom_collection, item_wrapper_class: 'custom-control', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
300
|
+
config.wrappers :custom_collection, item_wrapper_class: 'custom-control', item_label_class: 'custom-control-label', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
300
301
|
b.use :html5
|
301
302
|
b.optional :readonly
|
302
303
|
b.wrapper :legend_tag, tag: 'legend', class: 'col-form-label pt-0' do |ba|
|
@@ -308,7 +309,7 @@ SimpleForm.setup do |config|
|
|
308
309
|
end
|
309
310
|
|
310
311
|
# custom input for inline radio buttons and check boxes
|
311
|
-
config.wrappers :custom_collection_inline, item_wrapper_class: 'custom-control custom-control-inline', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
312
|
+
config.wrappers :custom_collection_inline, item_wrapper_class: 'custom-control custom-control-inline', item_label_class: 'custom-control-label', tag: 'fieldset', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
312
313
|
b.use :html5
|
313
314
|
b.optional :readonly
|
314
315
|
b.wrapper :legend_tag, tag: 'legend', class: 'col-form-label pt-0' do |ba|
|
@@ -326,7 +327,7 @@ SimpleForm.setup do |config|
|
|
326
327
|
b.optional :maxlength
|
327
328
|
b.optional :minlength
|
328
329
|
b.optional :readonly
|
329
|
-
b.use :label
|
330
|
+
b.use :label
|
330
331
|
b.wrapper :custom_file_wrapper, tag: 'div', class: 'custom-file' do |ba|
|
331
332
|
ba.use :input, class: 'custom-file-input', error_class: 'is-invalid', valid_class: 'is-valid'
|
332
333
|
ba.use :label, class: 'custom-file-label'
|
@@ -339,7 +340,7 @@ SimpleForm.setup do |config|
|
|
339
340
|
config.wrappers :custom_multi_select, tag: 'div', class: 'form-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
340
341
|
b.use :html5
|
341
342
|
b.optional :readonly
|
342
|
-
b.use :label
|
343
|
+
b.use :label
|
343
344
|
b.wrapper tag: 'div', class: 'd-flex flex-row justify-content-between align-items-center' do |ba|
|
344
345
|
ba.use :input, class: 'custom-select mx-1', error_class: 'is-invalid', valid_class: 'is-valid'
|
345
346
|
end
|
@@ -353,7 +354,7 @@ SimpleForm.setup do |config|
|
|
353
354
|
b.use :placeholder
|
354
355
|
b.optional :readonly
|
355
356
|
b.optional :step
|
356
|
-
b.use :label
|
357
|
+
b.use :label
|
357
358
|
b.use :input, class: 'custom-range', error_class: 'is-invalid', valid_class: 'is-valid'
|
358
359
|
b.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback d-block' }
|
359
360
|
b.use :hint, wrap_with: { tag: 'small', class: 'form-text text-muted' }
|
@@ -370,7 +371,7 @@ SimpleForm.setup do |config|
|
|
370
371
|
# b.optional :pattern
|
371
372
|
# b.optional :min_max
|
372
373
|
# b.optional :readonly
|
373
|
-
# b.use :label
|
374
|
+
# b.use :label
|
374
375
|
# b.wrapper :input_group_tag, tag: 'div', class: 'input-group' do |ba|
|
375
376
|
# ba.optional :prepend
|
376
377
|
# ba.use :input, class: 'form-control', error_class: 'is-invalid', valid_class: 'is-valid'
|
@@ -393,7 +394,7 @@ SimpleForm.setup do |config|
|
|
393
394
|
b.optional :min_max
|
394
395
|
b.optional :readonly
|
395
396
|
b.use :input, class: 'form-control', error_class: 'is-invalid', valid_class: 'is-valid'
|
396
|
-
b.use :label
|
397
|
+
b.use :label
|
397
398
|
b.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback' }
|
398
399
|
b.use :hint, wrap_with: { tag: 'small', class: 'form-text text-muted' }
|
399
400
|
end
|
@@ -402,8 +403,8 @@ SimpleForm.setup do |config|
|
|
402
403
|
config.wrappers :floating_labels_select, tag: 'div', class: 'form-label-group', error_class: 'form-group-invalid', valid_class: 'form-group-valid' do |b|
|
403
404
|
b.use :html5
|
404
405
|
b.optional :readonly
|
405
|
-
b.use :input, class: 'custom-select
|
406
|
-
b.use :label
|
406
|
+
b.use :input, class: 'custom-select', error_class: 'is-invalid', valid_class: 'is-valid'
|
407
|
+
b.use :label
|
407
408
|
b.use :full_error, wrap_with: { tag: 'div', class: 'invalid-feedback' }
|
408
409
|
b.use :hint, wrap_with: { tag: 'small', class: 'form-text text-muted' }
|
409
410
|
end
|
@@ -11,7 +11,7 @@ module SimpleForm
|
|
11
11
|
end
|
12
12
|
|
13
13
|
def has_errors?
|
14
|
-
|
14
|
+
object_with_errors? || object.nil? && has_custom_error?
|
15
15
|
end
|
16
16
|
|
17
17
|
def has_value?
|
@@ -34,6 +34,10 @@ module SimpleForm
|
|
34
34
|
has_custom_error? ? options[:error] : full_errors.send(error_method)
|
35
35
|
end
|
36
36
|
|
37
|
+
def object_with_errors?
|
38
|
+
object && object.respond_to?(:errors) && errors.present?
|
39
|
+
end
|
40
|
+
|
37
41
|
def error_method
|
38
42
|
options[:error_method] || SimpleForm.error_method
|
39
43
|
end
|
@@ -165,7 +165,7 @@ module SimpleForm
|
|
165
165
|
components = (wrapper.components.map(&:namespace) & ATTRIBUTE_COMPONENTS)
|
166
166
|
|
167
167
|
options = options.dup
|
168
|
-
options[:input_html] = options.except(:as, :boolean_style, :collection, :label_method, :value_method, :prompt, *components)
|
168
|
+
options[:input_html] = options.except(:as, :boolean_style, :collection, :disabled, :label_method, :value_method, :prompt, *components)
|
169
169
|
options = @defaults.deep_dup.deep_merge(options) if @defaults
|
170
170
|
|
171
171
|
input = find_input(attribute_name, options)
|
@@ -511,7 +511,7 @@ module SimpleForm
|
|
511
511
|
when :has_one
|
512
512
|
raise ArgumentError, ":has_one associations are not supported by f.association"
|
513
513
|
else
|
514
|
-
if options[:as] == :select
|
514
|
+
if options[:as] == :select || options[:as] == :grouped_select
|
515
515
|
html_options = options[:input_html] ||= {}
|
516
516
|
html_options[:multiple] = true unless html_options.key?(:multiple)
|
517
517
|
end
|
@@ -552,12 +552,12 @@ module SimpleForm
|
|
552
552
|
:datetime
|
553
553
|
when :string, :citext, nil
|
554
554
|
case attribute_name.to_s
|
555
|
-
when /password/ then :password
|
556
|
-
when /time_zone/ then :time_zone
|
557
|
-
when /country/ then :country
|
558
|
-
when /email/ then :email
|
559
|
-
when /phone/ then :tel
|
560
|
-
when /url/ then :url
|
555
|
+
when /(?:\b|\W|_)password(?:\b|\W|_)/ then :password
|
556
|
+
when /(?:\b|\W|_)time_zone(?:\b|\W|_)/ then :time_zone
|
557
|
+
when /(?:\b|\W|_)country(?:\b|\W|_)/ then :country
|
558
|
+
when /(?:\b|\W|_)email(?:\b|\W|_)/ then :email
|
559
|
+
when /(?:\b|\W|_)phone(?:\b|\W|_)/ then :tel
|
560
|
+
when /(?:\b|\W|_)url(?:\b|\W|_)/ then :url
|
561
561
|
else
|
562
562
|
file_method?(attribute_name) ? :file : (input_type || :string)
|
563
563
|
end
|
@@ -572,9 +572,28 @@ module SimpleForm
|
|
572
572
|
}.try(:last) if SimpleForm.input_mappings
|
573
573
|
end
|
574
574
|
|
575
|
+
# Internal: Try to discover whether an attribute corresponds to a file or not.
|
576
|
+
#
|
577
|
+
# Most upload Gems add some kind of attributes to the ActiveRecord's model they are included in.
|
578
|
+
# This method tries to guess if an attribute belongs to some of these Gems by checking the presence
|
579
|
+
# of their methods using `#respond_to?`.
|
580
|
+
#
|
581
|
+
# Note: This does not support multiple file upload inputs, as this is very application-specific.
|
582
|
+
#
|
583
|
+
# The order here was choosen based on the popularity of Gems and for commodity - e.g. the method
|
584
|
+
# with the suffix `_url` is present in three Gems, so it's checked with priority:
|
585
|
+
#
|
586
|
+
# - `#{attribute_name}_attachment` - ActiveStorage >= `5.2` and Refile >= `0.2.0` <= `0.4.0`
|
587
|
+
# - `#{attribute_name}_url` - Shrine >= `0.9.0`, Refile >= `0.6.0` and CarrierWave >= `0.2.1`
|
588
|
+
# - `#{attribute_name}_attacher` - Refile >= `0.4.0` and Shrine >= `0.9.0`
|
589
|
+
# - `#{attribute_name}_file_name` - Paperclip ~> `2.0` (added for backwards compatibility)
|
590
|
+
#
|
591
|
+
# Returns a Boolean.
|
575
592
|
def file_method?(attribute_name)
|
576
|
-
|
577
|
-
|
593
|
+
@object.respond_to?("#{attribute_name}_attachment") ||
|
594
|
+
@object.respond_to?("#{attribute_name}_url") ||
|
595
|
+
@object.respond_to?("#{attribute_name}_attacher") ||
|
596
|
+
@object.respond_to?("#{attribute_name}_file_name")
|
578
597
|
end
|
579
598
|
|
580
599
|
def find_attribute_column(attribute_name)
|
@@ -72,7 +72,10 @@ module SimpleForm
|
|
72
72
|
@html_classes = SimpleForm.additional_classes_for(:input) { additional_classes }
|
73
73
|
|
74
74
|
@input_html_classes = @html_classes.dup
|
75
|
-
|
75
|
+
|
76
|
+
input_html_classes = self.input_html_classes
|
77
|
+
|
78
|
+
if SimpleForm.input_class && input_html_classes.any?
|
76
79
|
input_html_classes << SimpleForm.input_class
|
77
80
|
end
|
78
81
|
|
@@ -63,6 +63,7 @@ module SimpleForm
|
|
63
63
|
return "" if !include_hidden? || !unchecked_value
|
64
64
|
options = { value: unchecked_value, id: nil, disabled: input_html_options[:disabled] }
|
65
65
|
options[:name] = input_html_options[:name] if input_html_options.key?(:name)
|
66
|
+
options[:form] = input_html_options[:form] if input_html_options.key?(:form)
|
66
67
|
|
67
68
|
@builder.hidden_field(attribute_name, options)
|
68
69
|
end
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module SimpleForm
|
3
|
+
module Inputs
|
4
|
+
class ColorInput < Base
|
5
|
+
def input(wrapper_options = nil)
|
6
|
+
input_html_options[:type] ||= "color" if html5?
|
7
|
+
|
8
|
+
merged_input_options = merge_wrapper_options(input_html_options, wrapper_options)
|
9
|
+
|
10
|
+
@builder.text_field(attribute_name, merged_input_options)
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
data/lib/simple_form/inputs.rb
CHANGED
data/lib/simple_form/tags.rb
CHANGED
@@ -48,7 +48,9 @@ module SimpleForm
|
|
48
48
|
private
|
49
49
|
|
50
50
|
def render_component(builder)
|
51
|
-
|
51
|
+
label_class = "#{@options[:item_label_class]} collection_radio_buttons".strip
|
52
|
+
|
53
|
+
builder.radio_button + builder.label(class: label_class)
|
52
54
|
end
|
53
55
|
end
|
54
56
|
|
@@ -62,7 +64,9 @@ module SimpleForm
|
|
62
64
|
private
|
63
65
|
|
64
66
|
def render_component(builder)
|
65
|
-
|
67
|
+
label_class = "#{@options[:item_label_class]} collection_check_boxes".strip
|
68
|
+
|
69
|
+
builder.check_box + builder.label(class: label_class)
|
66
70
|
end
|
67
71
|
end
|
68
72
|
end
|
data/lib/simple_form/version.rb
CHANGED
data/lib/simple_form.rb
CHANGED
@@ -38,6 +38,17 @@ to
|
|
38
38
|
See https://github.com/plataformatec/simple_form/pull/997 for more information.
|
39
39
|
WARN
|
40
40
|
|
41
|
+
FILE_METHODS_DEPRECATION_WARN = <<-WARN
|
42
|
+
[SIMPLE_FORM] SimpleForm.file_methods is deprecated and has no effect.
|
43
|
+
|
44
|
+
Since version 5, Simple Form now supports automatically discover of file inputs for the following Gems: activestorage, carrierwave, paperclip, refile and shrine.
|
45
|
+
If you are using a custom method that is not from one of the supported Gems, please change your forms to pass the input type explicitly:
|
46
|
+
|
47
|
+
<%= form.input :avatar, as: :file %>
|
48
|
+
|
49
|
+
See http://blog.plataformatec.com.br/2019/09/incorrect-access-control-in-simple-form-cve-2019-16676 for more information.
|
50
|
+
WARN
|
51
|
+
|
41
52
|
@@configured = false
|
42
53
|
|
43
54
|
def self.configured? #:nodoc:
|
@@ -120,10 +131,6 @@ See https://github.com/plataformatec/simple_form/pull/997 for more information.
|
|
120
131
|
mattr_accessor :browser_validations
|
121
132
|
@@browser_validations = true
|
122
133
|
|
123
|
-
# Collection of methods to detect if a file type was given.
|
124
|
-
mattr_accessor :file_methods
|
125
|
-
@@file_methods = %i[mounted_as file? public_filename attached?]
|
126
|
-
|
127
134
|
# Custom mappings for input types. This should be a hash containing a regexp
|
128
135
|
# to match as key, and the input type that will be used when the field name
|
129
136
|
# matches the regexp as value, such as { /count/ => :integer }.
|
@@ -265,6 +272,16 @@ See https://github.com/plataformatec/simple_form/pull/997 for more information.
|
|
265
272
|
@@form_class = value
|
266
273
|
end
|
267
274
|
|
275
|
+
def self.file_methods=(file_methods)
|
276
|
+
ActiveSupport::Deprecation.warn(FILE_METHODS_DEPRECATION_WARN, caller)
|
277
|
+
@@file_methods = file_methods
|
278
|
+
end
|
279
|
+
|
280
|
+
def self.file_methods
|
281
|
+
ActiveSupport::Deprecation.warn(FILE_METHODS_DEPRECATION_WARN, caller)
|
282
|
+
@@file_methods
|
283
|
+
end
|
284
|
+
|
268
285
|
# Default way to setup Simple Form. Run rails generate simple_form:install
|
269
286
|
# to create a fresh initializer with all configuration values.
|
270
287
|
def self.setup
|
@@ -45,8 +45,17 @@ class BuilderTest < ActionView::TestCase
|
|
45
45
|
|
46
46
|
test "collection radio sanitizes collection values for labels correctly" do
|
47
47
|
with_collection_radio_buttons @user, :name, ['$0.99', '$1.99'], :to_s, :to_s
|
48
|
-
|
49
|
-
|
48
|
+
|
49
|
+
# Rails 6 changed the way it sanitizes the values
|
50
|
+
# https://github.com/rails/rails/blob/6-0-stable/actionview/lib/action_view/helpers/tags/base.rb#L141
|
51
|
+
# https://github.com/rails/rails/blob/5-2-stable/actionview/lib/action_view/helpers/tags/base.rb#L141
|
52
|
+
if ActionView::VERSION::MAJOR == 5
|
53
|
+
assert_select 'label.collection_radio_buttons[for=user_name_099]', '$0.99'
|
54
|
+
assert_select 'label.collection_radio_buttons[for=user_name_199]', '$1.99'
|
55
|
+
else
|
56
|
+
assert_select 'label.collection_radio_buttons[for=user_name_0_99]', '$0.99'
|
57
|
+
assert_select 'label.collection_radio_buttons[for=user_name_1_99]', '$1.99'
|
58
|
+
end
|
50
59
|
end
|
51
60
|
|
52
61
|
test "collection radio checks the correct value to local variables" do
|
@@ -292,8 +301,17 @@ class BuilderTest < ActionView::TestCase
|
|
292
301
|
|
293
302
|
test "collection check box sanitizes collection values for labels correctly" do
|
294
303
|
with_collection_check_boxes @user, :name, ['$0.99', '$1.99'], :to_s, :to_s
|
295
|
-
|
296
|
-
|
304
|
+
|
305
|
+
# Rails 6 changed the way it sanitizes the values
|
306
|
+
# https://github.com/rails/rails/blob/6-0-stable/actionview/lib/action_view/helpers/tags/base.rb#L141
|
307
|
+
# https://github.com/rails/rails/blob/5-2-stable/actionview/lib/action_view/helpers/tags/base.rb#L141
|
308
|
+
if ActionView::VERSION::MAJOR == 5
|
309
|
+
assert_select 'label.collection_check_boxes[for=user_name_099]', '$0.99'
|
310
|
+
assert_select 'label.collection_check_boxes[for=user_name_199]', '$1.99'
|
311
|
+
else
|
312
|
+
assert_select 'label.collection_check_boxes[for=user_name_0_99]', '$0.99'
|
313
|
+
assert_select 'label.collection_check_boxes[for=user_name_1_99]', '$1.99'
|
314
|
+
end
|
297
315
|
end
|
298
316
|
|
299
317
|
test "collection check box checks the correct value to local variables" do
|
@@ -243,4 +243,10 @@ class AssociationTest < ActionView::TestCase
|
|
243
243
|
assert_equal({ as: :check_boxes, collection_wrapper_tag: :ul, item_wrapper_tag: :li },
|
244
244
|
options)
|
245
245
|
end
|
246
|
+
|
247
|
+
test 'builder with group select considers multiple select by default' do
|
248
|
+
with_association_for @user, :tags, as: :grouped_select, group_method: :group_method
|
249
|
+
|
250
|
+
assert_select 'select[multiple="multiple"].grouped_select'
|
251
|
+
end
|
246
252
|
end
|
@@ -224,6 +224,12 @@ class ErrorTest < ActionView::TestCase
|
|
224
224
|
assert_no_select 'span.error'
|
225
225
|
end
|
226
226
|
|
227
|
+
test 'input with custom error works when form does not use a model' do
|
228
|
+
with_form_for :user, :active, error: "Super User Active! cannot be blank"
|
229
|
+
|
230
|
+
assert_select 'span.error'
|
231
|
+
end
|
232
|
+
|
227
233
|
test 'input with custom error works when using full_error component' do
|
228
234
|
swap_wrapper :default, custom_wrapper_with_full_error do
|
229
235
|
error_text = "Super User Name! cannot be blank"
|