form_obj 0.3.0 → 0.4.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.
- checksums.yaml +4 -4
- data/.gitignore +2 -0
- data/.travis.yml +26 -1
- data/Appraisals +41 -0
- data/CHANGELOG.md +42 -0
- data/Gemfile.lock +51 -39
- data/README.md +140 -89
- data/bundle_install.sh +27 -0
- data/form_obj.gemspec +2 -1
- data/gemfiles/3.2.gemfile +9 -0
- data/gemfiles/4.0.gemfile +9 -0
- data/gemfiles/4.1.gemfile +9 -0
- data/gemfiles/4.2.gemfile +9 -0
- data/gemfiles/5.0.gemfile +9 -0
- data/gemfiles/5.1.gemfile +9 -0
- data/gemfiles/5.2.gemfile +9 -0
- data/install_rubies_and_bundler.sh +28 -0
- data/lib/form_obj/form/array.rb +36 -0
- data/lib/form_obj/form/attribute.rb +19 -0
- data/lib/form_obj/form/attributes.rb +13 -0
- data/lib/form_obj/form.rb +20 -26
- data/lib/form_obj/{mappable → model_mapper}/array.rb +5 -5
- data/lib/form_obj/{mappable → model_mapper}/attribute.rb +8 -9
- data/lib/form_obj/{mappable → model_mapper}/model_attribute/item.rb +1 -1
- data/lib/form_obj/{mappable → model_mapper}/model_attribute.rb +2 -2
- data/lib/form_obj/{mappable → model_mapper}/model_primary_key.rb +1 -1
- data/lib/form_obj/{mappable.rb → model_mapper.rb} +16 -15
- data/lib/form_obj/struct/array.rb +22 -0
- data/lib/form_obj/struct/attribute.rb +69 -0
- data/lib/form_obj/struct/attributes.rb +21 -0
- data/lib/form_obj/struct.rb +76 -0
- data/lib/form_obj/version.rb +1 -1
- data/lib/form_obj.rb +3 -2
- data/run_spec.sh +29 -0
- metadata +45 -16
- data/.ruby-version +0 -1
- data/lib/form_obj/array.rb +0 -36
- data/lib/form_obj/attribute.rb +0 -19
- data/lib/form_obj/attributes.rb +0 -13
data/README.md
CHANGED
@@ -1,7 +1,18 @@
|
|
1
1
|
# FormObj
|
2
2
|
|
3
|
-
|
4
|
-
|
3
|
+
[](https://badge.fury.io/rb/form_obj)
|
4
|
+
[](https://travis-ci.com/akoltun/form_obj)
|
5
|
+
|
6
|
+
Form Object allows to define a complicated data structure (using nesting, arrays) and use it with Rails-compatible form builders.
|
7
|
+
A Form Object can be serialized and deserialized to a model and/or a hash.
|
8
|
+
|
9
|
+
## Compatibility and Dependency Requirements
|
10
|
+
|
11
|
+
Ruby: 2.2.8+
|
12
|
+
ActiveSupport: 3.2+
|
13
|
+
ActiveModel: 3.2+
|
14
|
+
|
15
|
+
The gem is tested against all ruby versions and all versions of its dependencies.
|
5
16
|
|
6
17
|
## Installation
|
7
18
|
|
@@ -21,16 +32,19 @@ Or install it yourself as:
|
|
21
32
|
|
22
33
|
## Usage
|
23
34
|
|
24
|
-
**WARNING!!!** The gem is still under development.
|
35
|
+
**WARNING!!!** The gem is still under development. Expect braking changes.<br/>
|
25
36
|
|
26
|
-
|
27
|
-
So on top of all `TreeStruct` functionality `FormObj::Obj` adds `update_attributes` method for mass update of attributes
|
28
|
-
(similar to ActuveRecord) and syntax sugar to easily use ActiveModel::Validations and ActiveModel::Errors with `TreeStruct`.
|
37
|
+
Class `FormObj::Struct` allows to describe complicated data structure.
|
29
38
|
|
30
|
-
`
|
31
|
-
|
32
|
-
|
33
|
-
|
39
|
+
Class `FormObj::Form` inherits from `FormObj::Struct` and adds form builder compatibility and includes ActiveModel validations.
|
40
|
+
|
41
|
+
Module `FormObj::ModelMapper` being included into `FormObj::Form` descendants allows to map a form object to a model
|
42
|
+
in order to be able to exchange attributes value between them:
|
43
|
+
* load attributes value from the model,
|
44
|
+
* sync attributes value to the model,
|
45
|
+
* represent a form object as a model hash (similar to the `to_hash` method but using the
|
46
|
+
model attributes name) and
|
47
|
+
* copy errors from the model(s) to the from object.
|
34
48
|
|
35
49
|
### Table of Contents
|
36
50
|
|
@@ -68,7 +82,7 @@ class SimpleForm < FormObj::Form
|
|
68
82
|
end
|
69
83
|
```
|
70
84
|
|
71
|
-
Use it in form builder.
|
85
|
+
Use it in a form builder.
|
72
86
|
|
73
87
|
```erb
|
74
88
|
<%= form_for(@simple_form) do |f| %>
|
@@ -89,7 +103,7 @@ class NestedForm < FormObj::Form
|
|
89
103
|
attribute :name
|
90
104
|
attribute :year
|
91
105
|
attribute :car do
|
92
|
-
attribute :
|
106
|
+
attribute :code
|
93
107
|
attribute :driver
|
94
108
|
attribute :engine do
|
95
109
|
attribute :power
|
@@ -107,7 +121,7 @@ class EngineForm < FormObj::Form
|
|
107
121
|
attribute :volume
|
108
122
|
end
|
109
123
|
class CarForm < FormObj::Form
|
110
|
-
attribute :
|
124
|
+
attribute :code
|
111
125
|
attribute :driver
|
112
126
|
attribute :engine, class: EngineForm
|
113
127
|
end
|
@@ -120,7 +134,7 @@ end
|
|
120
134
|
|
121
135
|
Use nested forms in form builder.
|
122
136
|
|
123
|
-
```
|
137
|
+
```erb
|
124
138
|
<%= form_for(@nested_form) do |f| %>
|
125
139
|
<%= f.label :name %>
|
126
140
|
<%= f.text_field :name %>
|
@@ -129,8 +143,8 @@ Use nested forms in form builder.
|
|
129
143
|
<%= f.text_field :year %>
|
130
144
|
|
131
145
|
<%= f.fields_for(:car) do |fc| %>
|
132
|
-
<%= fc.label :
|
133
|
-
<%= fc.text_field :
|
146
|
+
<%= fc.label :code %>
|
147
|
+
<%= fc.text_field :code %>
|
134
148
|
|
135
149
|
<%= fc.label :driver %>
|
136
150
|
<%= fc.text_field :driver %>
|
@@ -155,7 +169,7 @@ class ArrayForm < FormObj::Form
|
|
155
169
|
attribute :name
|
156
170
|
attribute :year
|
157
171
|
attribute :cars, array: true do
|
158
|
-
attribute :
|
172
|
+
attribute :code
|
159
173
|
attribute :driver
|
160
174
|
attribute :engine do
|
161
175
|
attribute :power
|
@@ -173,7 +187,7 @@ class EngineForm < FormObj::Form
|
|
173
187
|
attribute :volume
|
174
188
|
end
|
175
189
|
class CarForm < FormObj::Form
|
176
|
-
attribute :
|
190
|
+
attribute :code
|
177
191
|
attribute :driver
|
178
192
|
attribute :engine, class: EngineForm
|
179
193
|
end
|
@@ -184,19 +198,55 @@ class ArrayForm < FormObj::Form
|
|
184
198
|
end
|
185
199
|
```
|
186
200
|
|
187
|
-
Add new elements in the array by using method :create
|
201
|
+
Add new elements in the array by using method :create.
|
188
202
|
|
189
203
|
```ruby
|
190
|
-
array_form = ArrayForm.new
|
191
|
-
array_form.size # => 0
|
192
|
-
array_form.cars.create
|
193
|
-
array_form.size # => 1
|
204
|
+
@array_form = ArrayForm.new
|
205
|
+
@array_form.cars.size # => 0
|
206
|
+
@array_form.cars.create
|
207
|
+
@array_form.cars.size # => 1
|
208
|
+
```
|
209
|
+
|
210
|
+
Use array of nested forms in the form builder.
|
211
|
+
|
212
|
+
```erb
|
213
|
+
<%= form_for(@array_form) do |f| %>
|
214
|
+
<%= f.label :name %>
|
215
|
+
<%= f.text_field :name %>
|
216
|
+
|
217
|
+
<%= f.label :year %>
|
218
|
+
<%= f.text_field :year %>
|
219
|
+
|
220
|
+
<% f.cars.each do |car| %>
|
221
|
+
<%= f.fields_for(:cars, car, index: '') do |fc| %>
|
222
|
+
<%= fc.label :code %>
|
223
|
+
<%= fc.text_field :code %>
|
224
|
+
|
225
|
+
<%= fc.label :driver %>
|
226
|
+
<%= fc.text_field :driver %>
|
227
|
+
|
228
|
+
<%= fc.field_for(:engine) do |fce| %>
|
229
|
+
<%= fce.label :power %>
|
230
|
+
<%= fce.text_field :power %>
|
231
|
+
|
232
|
+
<%= fce.label :volume %>
|
233
|
+
<%= fce.text_field :volume %>
|
234
|
+
<% end %>
|
235
|
+
<% end %>
|
236
|
+
<% end %>
|
237
|
+
<% end %>
|
194
238
|
```
|
195
239
|
|
196
240
|
### 2. Update Attributes
|
197
241
|
|
198
242
|
Update form object attributes with the parameter hash received from the browser.
|
199
|
-
Method `update_attributes(new_attrs_hash)` returns self so one can chain calls.
|
243
|
+
Method `update_attributes(new_attrs_hash, options)` returns self so one can chain calls.
|
244
|
+
|
245
|
+
`options` hash can have `:raise_if_not_found` key which has `true` value by default.
|
246
|
+
If `new_attrs_hash` has key that does not correspond to any attributes
|
247
|
+
and `raise_if_not_found` is `true` than `UnknownAttributeError` will be generated.
|
248
|
+
`raise_if_not_found` equals to `false` prevents error generation
|
249
|
+
and non existent attribute will be just ignored.
|
200
250
|
|
201
251
|
```ruby
|
202
252
|
simple_form = SimpleForm.new
|
@@ -216,7 +266,7 @@ simple_form.year # => 1966
|
|
216
266
|
nested_form = NestedForm.new
|
217
267
|
nested_form.name = 'Ferrari'
|
218
268
|
nested_form.year = 1950
|
219
|
-
nested_form.car.
|
269
|
+
nested_form.car.code = '340 F1'
|
220
270
|
nested_form.car.driver = 'Ascari'
|
221
271
|
nested_form.car.engine.power = 335
|
222
272
|
nested_form.car.engine.volume = 4.1
|
@@ -224,7 +274,7 @@ nested_form.update_attributes(
|
|
224
274
|
name: 'McLaren',
|
225
275
|
year: 1966,
|
226
276
|
car: {
|
227
|
-
|
277
|
+
code: 'M2B',
|
228
278
|
driver: 'Bruce McLaren',
|
229
279
|
engine: {
|
230
280
|
power: 300,
|
@@ -234,7 +284,7 @@ nested_form.update_attributes(
|
|
234
284
|
)
|
235
285
|
nested_form.name # => "McLaren"
|
236
286
|
nested_form.year # => 1966
|
237
|
-
nested_form.car.
|
287
|
+
nested_form.car.code # => "M2B"
|
238
288
|
nested_form.car.driver # => "Bruce McLaren"
|
239
289
|
nested_form.car.engine.power # => 300
|
240
290
|
nested_form.car.engine.volume # => 3.0
|
@@ -242,20 +292,20 @@ nested_form.car.engine.volume # => 3.0
|
|
242
292
|
|
243
293
|
#### 2.2. Array of Form Objects
|
244
294
|
|
245
|
-
Updating array of form objects will compare the existing array
|
295
|
+
Updating an array of form objects will compare the existing array with the new one.
|
246
296
|
New array elements will be added, existing array elements will be updated, absent array elements will be deleted
|
247
|
-
(deleting
|
297
|
+
(deleting behavior is the subject of changes in future releases - only elements with flag `_destroy == true` will be deleted).
|
248
298
|
|
249
299
|
In order to compare old and new array its elements have to be identified via the primary key.
|
250
300
|
Primary key can be specified either on the attribute level or on the form level.
|
251
|
-
If it is not specified the :id field is supposed to be
|
301
|
+
If it is not specified the :id field is supposed to be the primary key.
|
252
302
|
|
253
303
|
```ruby
|
254
304
|
class ArrayForm < FormObj::Form
|
255
305
|
attribute :name
|
256
306
|
attribute :year
|
257
307
|
attribute :cars, array: true do
|
258
|
-
attribute :
|
308
|
+
attribute :code, primary_key: true # <- primary key is specified on attribute level
|
259
309
|
attribute :driver
|
260
310
|
attribute :engine do
|
261
311
|
attribute :power
|
@@ -269,8 +319,8 @@ end
|
|
269
319
|
class ArrayForm < FormObj::Form
|
270
320
|
attribute :name
|
271
321
|
attribute :year
|
272
|
-
attribute :cars, array: true, primary_key: :
|
273
|
-
attribute :
|
322
|
+
attribute :cars, array: true, primary_key: :code do # <- primary key is specified on form level
|
323
|
+
attribute :code
|
274
324
|
attribute :driver
|
275
325
|
attribute :engine do
|
276
326
|
attribute :power
|
@@ -286,13 +336,13 @@ array_form.name = 'Ferrari'
|
|
286
336
|
array_form.year = 1950
|
287
337
|
|
288
338
|
car1 = array_form.cars.create
|
289
|
-
car1.
|
339
|
+
car1.code = '340 F1'
|
290
340
|
car1.driver = 'Ascari'
|
291
341
|
car1.engine.power = 335
|
292
342
|
car1.engine.volume = 4.1
|
293
343
|
|
294
344
|
car2 = array_form.cars.create
|
295
|
-
car2.
|
345
|
+
car2.code = 'M2B'
|
296
346
|
car2.driver = 'Villoresi'
|
297
347
|
car2.engine.power = 300
|
298
348
|
car2.engine.volume = 3.3
|
@@ -302,13 +352,13 @@ array_form.update_attributes(
|
|
302
352
|
year: 1966,
|
303
353
|
cars: [
|
304
354
|
{
|
305
|
-
|
355
|
+
code: 'M2B',
|
306
356
|
driver: 'Bruce McLaren',
|
307
357
|
engine: {
|
308
358
|
volume: 3.0
|
309
359
|
}
|
310
360
|
}, {
|
311
|
-
|
361
|
+
code: 'M7A',
|
312
362
|
driver: 'Denis Hulme',
|
313
363
|
engine: {
|
314
364
|
power: 415,
|
@@ -320,12 +370,12 @@ array_form.update_attributes(
|
|
320
370
|
array_form.name # => "McLaren"
|
321
371
|
array_form.year # => 1966
|
322
372
|
|
323
|
-
array_form.cars[0].
|
373
|
+
array_form.cars[0].code # => "M2B"
|
324
374
|
array_form.cars[0].driver # => "Bruce McLaren"
|
325
375
|
array_form.cars[0].engine.power # => 300 - this value was not updated in update_attributes
|
326
376
|
array_form.cars[0].engine.volume # => 3.0
|
327
377
|
|
328
|
-
array_form.cars[1].
|
378
|
+
array_form.cars[1].code # => "M7A"
|
329
379
|
array_form.cars[1].driver # => "Denis Hulme"
|
330
380
|
array_form.cars[1].engine.power # => 415
|
331
381
|
array_form.cars[1].engine.volume # => nil - this value is nil because this car was created in updated_attributes
|
@@ -333,7 +383,7 @@ array_form.cars[1].engine.volume # => nil - this value is nil because this
|
|
333
383
|
|
334
384
|
### 3. Serialize to Hash
|
335
385
|
|
336
|
-
Call `to_hash()` method in order to get hash representation of the form object
|
386
|
+
Call `to_hash()` method in order to get a hash representation of the form object
|
337
387
|
|
338
388
|
```ruby
|
339
389
|
simple_form.to_hash # => {
|
@@ -349,7 +399,7 @@ nested_form.to_hash # => {
|
|
349
399
|
# => :name => "McLaren",
|
350
400
|
# => :year => 1966,
|
351
401
|
# => :car => {
|
352
|
-
# => :
|
402
|
+
# => :code => "340 F1",
|
353
403
|
# => :driver => "Ascari",
|
354
404
|
# => :engine => {
|
355
405
|
# => :power => 335,
|
@@ -366,14 +416,14 @@ array_form.to_hash # => {
|
|
366
416
|
# => :name => "McLaren",
|
367
417
|
# => :year => 1966,
|
368
418
|
# => :cars => [{
|
369
|
-
# => :
|
419
|
+
# => :code => "M2B",
|
370
420
|
# => :driver => "Bruce McLaren",
|
371
421
|
# => :engine => {
|
372
422
|
# => :power => 300,
|
373
423
|
# => :volume => 3.0
|
374
424
|
# => }
|
375
425
|
# => }, {
|
376
|
-
# => :
|
426
|
+
# => :code => "M7A",
|
377
427
|
# => :driver => "Denis Hulme",
|
378
428
|
# => :engine => {
|
379
429
|
# => :power => 415,
|
@@ -383,16 +433,16 @@ array_form.to_hash # => {
|
|
383
433
|
# => }
|
384
434
|
```
|
385
435
|
|
386
|
-
### 4. Map Form Object to Models
|
436
|
+
### 4. Map Form Object to/from Models
|
387
437
|
|
388
|
-
Include `
|
438
|
+
Include `DataMapper` mix-in and map form object attributes to one or more models by using `:model` and `:model_attribute` parameters.
|
389
439
|
By default each form object attribute is mapped to the model attribute with the same name of the `:default` model.
|
390
440
|
|
391
|
-
Use dot notation to map model attribute to nested model. Use colon to specify "hash" attribute.
|
441
|
+
Use dot notation to map model attribute to a nested model. Use colon to specify a "hash" attribute.
|
392
442
|
|
393
443
|
```ruby
|
394
444
|
class SingleForm < FormObj::Form
|
395
|
-
include
|
445
|
+
include DataMapper
|
396
446
|
|
397
447
|
attribute :name, model_attribute: :team_name
|
398
448
|
attribute :year
|
@@ -412,7 +462,7 @@ Suppose `single_form = SingleForm.new` and `model` to be an instance of a model.
|
|
412
462
|
|
413
463
|
```ruby
|
414
464
|
class MultiForm < FormObj::Form
|
415
|
-
include
|
465
|
+
include ModelMapper
|
416
466
|
|
417
467
|
attribute :name, model_attribute: :team_name
|
418
468
|
attribute :year
|
@@ -434,7 +484,7 @@ Use `model_attribute: false` in order to avoid attribute mapping to the model.
|
|
434
484
|
|
435
485
|
```ruby
|
436
486
|
class SimpleForm < FormObj::Form
|
437
|
-
include
|
487
|
+
include ModelMapper
|
438
488
|
|
439
489
|
attribute :name, model_attribute: :team_name
|
440
490
|
attribute :year
|
@@ -452,16 +502,17 @@ Suppose `form = SimpleForm.new` and `model` to be an instance of a model.
|
|
452
502
|
|
453
503
|
##### 4.2.1. Map Nested Form Object Attribute to Parent Level Model Attribute
|
454
504
|
|
505
|
+
TODO: replace `model_attribute` by `model`
|
455
506
|
Use `model_attribute: false` for nested form object in order to map its attributes to the parent level of the model.
|
456
507
|
|
457
508
|
```ruby
|
458
509
|
class NestedForm < FormObj::Form
|
459
|
-
include
|
510
|
+
include ModelMapper
|
460
511
|
|
461
512
|
attribute :name, model_attribute: :team_name
|
462
513
|
attribute :year
|
463
514
|
attribute :car, model_attribute: false do # nesting only in form object but not in a model
|
464
|
-
attribute :
|
515
|
+
attribute :code
|
465
516
|
attribute :driver
|
466
517
|
attribute :engine do
|
467
518
|
attribute :power
|
@@ -477,23 +528,23 @@ Suppose `form = NestedForm.new` and `model` to be an instance of a model.
|
|
477
528
|
| --------------------- | --------------- |
|
478
529
|
| `form.name` | `model.team_name` |
|
479
530
|
| `form.year` | `model.year` |
|
480
|
-
| `form.car.
|
531
|
+
| `form.car.code` | `model.code` |
|
481
532
|
| `form.car.driver` | `model.driver` |
|
482
533
|
| `form.car.engine.power` | `model.engine.power` |
|
483
534
|
| `form.car.engine.volume` | `model.engine.volume` |
|
484
535
|
|
485
536
|
#### 4.3. Map Nested Form Object to A Hash Model
|
486
537
|
|
487
|
-
Use `
|
538
|
+
Use `model_hash: true` in order to map a nested form object to a hash as a model.
|
488
539
|
|
489
540
|
```ruby
|
490
541
|
class NestedForm < FormObj::Form
|
491
|
-
include
|
542
|
+
include ModelMapper
|
492
543
|
|
493
544
|
attribute :name, model_attribute: :team_name
|
494
545
|
attribute :year
|
495
|
-
attribute :car,
|
496
|
-
attribute :
|
546
|
+
attribute :car, model_hash: true do # nesting only in form object but not in a model
|
547
|
+
attribute :code
|
497
548
|
attribute :driver
|
498
549
|
attribute :engine do
|
499
550
|
attribute :power
|
@@ -509,7 +560,7 @@ Suppose `form = NestedForm.new` and `model` to be an instance of a model.
|
|
509
560
|
| --------------------- | --------------- |
|
510
561
|
| `form.name` | `model.team_name` |
|
511
562
|
| `form.year` | `model.year` |
|
512
|
-
| `form.car.
|
563
|
+
| `form.car.code` | `model.car[:code]` |
|
513
564
|
| `form.car.driver` | `model.car[:driver]` |
|
514
565
|
| `form.car.engine.power` | `model.car[:engine].power` |
|
515
566
|
| `form.car.engine.volume` | `model.car[:engine].volume` |
|
@@ -521,7 +572,7 @@ Method returns self so one can chain calls.
|
|
521
572
|
|
522
573
|
```ruby
|
523
574
|
class MultiForm < FormObj::Form
|
524
|
-
include
|
575
|
+
include ModelMapper
|
525
576
|
|
526
577
|
attribute :name, model_attribute: :team_name
|
527
578
|
attribute :year
|
@@ -541,14 +592,14 @@ multi_form.to_hash # => {
|
|
541
592
|
|
542
593
|
Use `load_from_models(default: model)` or `load_from_model(model)` to load from single model.
|
543
594
|
|
544
|
-
### 6.
|
595
|
+
### 6. Sync Form Object to Models
|
545
596
|
|
546
|
-
Use `
|
597
|
+
Use `sync_to_models(models)` to sync form object attributes to mapped models.
|
547
598
|
Method returns self so one can chain calls.
|
548
599
|
|
549
600
|
```ruby
|
550
601
|
class MultiForm < FormObj::Form
|
551
|
-
include
|
602
|
+
include ModelMapper
|
552
603
|
|
553
604
|
attribute :name, model_attribute: :team_name
|
554
605
|
attribute :year
|
@@ -560,18 +611,18 @@ car_model = { engine: Struct.new(:power).new(335) }
|
|
560
611
|
|
561
612
|
multi_form = MultiForm.new
|
562
613
|
multi_form.update_attributes(name: 'McLaren', year: 1966, engine_power: 415)
|
563
|
-
multi_form.
|
614
|
+
multi_form.sync_to_models(default: default_model, car: car_model)
|
564
615
|
|
565
616
|
default_model.name # => "McLaren"
|
566
617
|
default_model.year # => 1966
|
567
618
|
car_model[:engine].power # => 415
|
568
619
|
```
|
569
620
|
|
570
|
-
Use `
|
621
|
+
Use `sync_to_models(default: model)` or `sync_to_model(model)` to sync to single model.
|
571
622
|
|
572
|
-
Neither `
|
623
|
+
Neither `sync_to_models` nor `sync_to_model` calls `save` method on the model(s).
|
573
624
|
Also they don't call `valid?` method on the model(s).
|
574
|
-
Instead they just assign form object attributes
|
625
|
+
Instead they just assign form object attributes value to mapped model attributes
|
575
626
|
using `<attribute_name>=` accessors on the model(s).
|
576
627
|
|
577
628
|
It is completely up to developer to do any additional validations on the model(s) and save it(them).
|
@@ -585,12 +636,12 @@ Form object will try to guess the name of the class from the name of the attribu
|
|
585
636
|
|
586
637
|
```ruby
|
587
638
|
class ArrayForm < FormObj::Form
|
588
|
-
include
|
639
|
+
include ModelMapper
|
589
640
|
|
590
641
|
attribute :name
|
591
642
|
attribute :year
|
592
643
|
attribute :cars, array: true, model_class: Car do
|
593
|
-
attribute :
|
644
|
+
attribute :code, primary_key: true # <- primary key is specified on attribute level
|
594
645
|
attribute :driver
|
595
646
|
end
|
596
647
|
end
|
@@ -601,12 +652,12 @@ nested models the value of `:model_class` parameter should be an array of corres
|
|
601
652
|
|
602
653
|
```ruby
|
603
654
|
class ArrayForm < FormObj::Form
|
604
|
-
include
|
655
|
+
include ModelMapper
|
605
656
|
|
606
657
|
attribute :name
|
607
658
|
attribute :year
|
608
659
|
attribute :cars, array: true, model_attribute: 'equipment.cars', model_class: [Equipment, Car] do
|
609
|
-
attribute :
|
660
|
+
attribute :code, primary_key: true # <- primary key is specified on attribute level
|
610
661
|
attribute :driver
|
611
662
|
end
|
612
663
|
end
|
@@ -618,7 +669,7 @@ Use `to_model_hash(model = :default)` to get hash representation of the model th
|
|
618
669
|
|
619
670
|
```ruby
|
620
671
|
class MultiForm < FormObj::Form
|
621
|
-
include
|
672
|
+
include ModelMapper
|
622
673
|
|
623
674
|
attribute :name, model_attribute: :team_name
|
624
675
|
attribute :year
|
@@ -646,12 +697,12 @@ If array of form objects mapped to the parent model (`model_attribute: false`) i
|
|
646
697
|
|
647
698
|
```ruby
|
648
699
|
class ArrayForm < FormObj::Form
|
649
|
-
include
|
700
|
+
include ModelMapper
|
650
701
|
|
651
702
|
attribute :name
|
652
703
|
attribute :year
|
653
704
|
attribute :cars, array: true, model_attribute: false do
|
654
|
-
attribute :
|
705
|
+
attribute :code, primary_key: true
|
655
706
|
attribute :driver
|
656
707
|
end
|
657
708
|
end
|
@@ -661,10 +712,10 @@ array_form.update_attributes(
|
|
661
712
|
name: 'McLaren',
|
662
713
|
year: 1966,
|
663
714
|
cars: [{
|
664
|
-
|
715
|
+
code: 'M2B',
|
665
716
|
driver: 'Bruce McLaren'
|
666
717
|
}, {
|
667
|
-
|
718
|
+
code: 'M7A',
|
668
719
|
driver: 'Denis Hulme'
|
669
720
|
}]
|
670
721
|
)
|
@@ -673,10 +724,10 @@ array_form.to_model_hash # => {
|
|
673
724
|
# => :team_name => "McLaren",
|
674
725
|
# => :year => 1966,
|
675
726
|
# => :self => {
|
676
|
-
# => :
|
727
|
+
# => :code => "M2B",
|
677
728
|
# => :driver => "Bruce McLaren"
|
678
729
|
# => }, {
|
679
|
-
# => :
|
730
|
+
# => :code => "M7A",
|
680
731
|
# => :driver => "Denis Hulme"
|
681
732
|
# => }
|
682
733
|
# => }
|
@@ -698,7 +749,7 @@ So add ActiveModel validations directly to Form Object class definition.
|
|
698
749
|
|
699
750
|
```ruby
|
700
751
|
class MultiForm < FormObj::Form
|
701
|
-
include
|
752
|
+
include ModelMapper
|
702
753
|
|
703
754
|
attribute :name, model_attribute: :team_name
|
704
755
|
attribute :year
|
@@ -757,7 +808,7 @@ class CreateCar < ActiveRecord::Migration
|
|
757
808
|
def change
|
758
809
|
create_table :cars do |t|
|
759
810
|
t.references :team
|
760
|
-
t.string :
|
811
|
+
t.string :code
|
761
812
|
t.text :engine
|
762
813
|
end
|
763
814
|
end
|
@@ -774,17 +825,17 @@ end
|
|
774
825
|
```ruby
|
775
826
|
# app/form_objects/team_form.rb
|
776
827
|
class TeamForm < FormObj::Form
|
777
|
-
include
|
828
|
+
include ModelMapper
|
778
829
|
|
779
830
|
attribute :id
|
780
831
|
attribute :name, model_attribute: :team_name
|
781
832
|
attribute :year
|
782
833
|
attribute :cars, array: true do
|
783
834
|
attribute :id
|
784
|
-
attribute :
|
835
|
+
attribute :code
|
785
836
|
attribute :engine_power, model_attribute: 'engine.:power'
|
786
837
|
|
787
|
-
validates :
|
838
|
+
validates :code, presence: true
|
788
839
|
end
|
789
840
|
|
790
841
|
validates :name, :year, presence: true
|
@@ -809,7 +860,7 @@ class TeamsController < ApplicationController
|
|
809
860
|
@team = TeamForm.new.update_attributes(params[:team])
|
810
861
|
|
811
862
|
if @team.valid?
|
812
|
-
@team.
|
863
|
+
@team.sync_to_model(model = Team.new)
|
813
864
|
if model.save
|
814
865
|
return redirect_to team_path(model), notice: 'Team has been created'
|
815
866
|
else
|
@@ -825,7 +876,7 @@ class TeamsController < ApplicationController
|
|
825
876
|
@team.update_attributes(params[:team])
|
826
877
|
|
827
878
|
if @team.valid?
|
828
|
-
@team.
|
879
|
+
@team.sync_to_model(model)
|
829
880
|
if model.save
|
830
881
|
return redirect_to team_path(model), notice: 'Team has been updated'
|
831
882
|
else
|
@@ -844,7 +895,7 @@ end
|
|
844
895
|
<p>Cars:</p>
|
845
896
|
<ul>
|
846
897
|
<% @team.cars.each do |car| %>
|
847
|
-
<li><%= car.
|
898
|
+
<li><%= car.code %> (<%= car.engine[:power] %> hp)</li>
|
848
899
|
<% end %>
|
849
900
|
</ul>
|
850
901
|
```
|
@@ -864,7 +915,7 @@ end
|
|
864
915
|
<%= f.text_field :year %>
|
865
916
|
|
866
917
|
<%= f.fields_for :cars do |cf| %>
|
867
|
-
<%= cf.text_field :
|
918
|
+
<%= cf.text_field :code %>
|
868
919
|
<%= cf.link_to_remove 'Remove the Car' %>
|
869
920
|
<% end %>
|
870
921
|
<%= f.link_to_add 'Add a Car', :cars %>
|
@@ -877,10 +928,10 @@ end
|
|
877
928
|
| --- |:---:|:---:|:---:| --- |
|
878
929
|
| array | block* or `:class`** | `false` | `TreeStruct` | This attribute is an array of form objects. The structure of array element form object is described either in the block or in the separate class referenced by `:class` parameter |
|
879
930
|
| class | - | - | `TreeStruct` | This attribute is either nested form object or array of form objects. The value of this parameter is the class of this form object or the name of the class. |
|
880
|
-
|
|
881
|
-
| model | - | `:default` | `FormObj::
|
882
|
-
| model_attribute | - | `<attribute_name>` | `FormObj::
|
883
|
-
| model_class | block* or `:class`** or dot notation for `:model_attribute`*** | `<attribute_name>.classify` | `FormObj::
|
931
|
+
| model_hash | block* or `:class`** | `false` | `FormObj::ModelMapper` | This attribute is either nested form object or array of form objects. This form object is mapped to a model of the class `Hash` so all its attributes should be accessed by `[:<attribute_name>]` instead of `.<attribute_name>` |
|
932
|
+
| model | - | `:default` | `FormObj::ModelMapper` | The name of the model to which this attribute is mapped |
|
933
|
+
| model_attribute | - | `<attribute_name>` | `FormObj::ModelMapper` | The name of the model attribute to which this form object attribute is mapped. Dot notation is used in order to map to nested model, ex. `"car.engine.power"`. Colon is used in front of the name if the model is hash, ex. `"car.:engine.power"` - means call to `#car` returns `Hash` so the model attribute should be accessed like `car[:engine].power`. `false` value means that attribute is not mapped. If attribute describes nested form object and has `model_attribute: false` the attributes of nested form will be called on the parent (upper level) model. If attribute describes array of form objects and has `model_attribute: false` the methods to access array elements (`:[]` etc.) will be called on the parent (upper level) model. |
|
934
|
+
| model_class | block* or `:class`** or dot notation for `:model_attribute`*** | `<attribute_name>.classify` | `FormObj::ModelMapper` | The class (or the name of the class) of the mapped model. |
|
884
935
|
| primary_key | no block* and no `:class`** | `false` | `FormObj::Form` | This attribute is the primary key of the form object. The mapped model attribute is considered to be a primary key for the corresponding model. |
|
885
936
|
| primary_key | block* or `:class`** | - | `FormObj::Form` | This attribute is either nested form object or array of form objects. The value of this parameter is the name of the primary key attribute of this form object. |
|
886
937
|
\* block - means that there is block definition for the attribute
|
data/bundle_install.sh
ADDED
@@ -0,0 +1,27 @@
|
|
1
|
+
#!/usr/bin/env bash
|
2
|
+
|
3
|
+
set -e
|
4
|
+
|
5
|
+
rubies=()
|
6
|
+
versions=false
|
7
|
+
while read -r line; do
|
8
|
+
if [[ $line == "rvm:" ]]
|
9
|
+
then
|
10
|
+
versions=true
|
11
|
+
elif $versions && [[ $line == -* ]]
|
12
|
+
then
|
13
|
+
rubies+=(${line:2})
|
14
|
+
elif [[ ${line:0:1} != "#" ]]
|
15
|
+
then
|
16
|
+
versions=false
|
17
|
+
fi
|
18
|
+
done < ".travis.yml"
|
19
|
+
|
20
|
+
for i in "${rubies[@]}"
|
21
|
+
do
|
22
|
+
echo "====================================================="
|
23
|
+
echo "$i: Start appraisal install"
|
24
|
+
echo "====================================================="
|
25
|
+
rvm $i exec bundle
|
26
|
+
rvm $i exec appraisal install
|
27
|
+
done
|
data/form_obj.gemspec
CHANGED
@@ -33,7 +33,7 @@ well as serialized to a hash which reflects a model. ActiveModel::Errors could b
|
|
33
33
|
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
34
34
|
spec.require_paths = ["lib"]
|
35
35
|
|
36
|
-
spec.add_dependency "
|
36
|
+
spec.add_dependency "typed_array", ">= 1.0.1"
|
37
37
|
spec.add_dependency "activemodel", ">= 3.2"
|
38
38
|
spec.add_dependency "activesupport", ">= 3.2"
|
39
39
|
|
@@ -41,4 +41,5 @@ well as serialized to a hash which reflects a model. ActiveModel::Errors could b
|
|
41
41
|
spec.add_development_dependency "rake", "~> 10.0"
|
42
42
|
spec.add_development_dependency "rspec", "~> 3.0"
|
43
43
|
spec.add_development_dependency "actionpack", ">= 3.2"
|
44
|
+
spec.add_development_dependency "appraisal"
|
44
45
|
end
|