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.
data/README.md CHANGED
@@ -1,7 +1,18 @@
1
1
  # FormObj
2
2
 
3
- Form Object allows to describe complicated data structure (nesting, arrays) and use it with Rails-cmpatible form builders.
4
- Form Object can serialize and deserialize itself to/from model and hash.
3
+ [![Gem Version](https://badge.fury.io/rb/form_obj.svg)](https://badge.fury.io/rb/form_obj)
4
+ [![Build Status](https://travis-ci.com/akoltun/form_obj.svg?branch=master)](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. Expecting braking changes.<br/>
35
+ **WARNING!!!** The gem is still under development. Expect braking changes.<br/>
25
36
 
26
- Form Object `FormObj::Form` is inherited from `TreeStruct` (https://github.com/akoltun/tree_struct).
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
- `Mappable` module included in `FormObj::Form` oblect allows to map form object to a model,
31
- load attributes from and attributes to it, represent form object as model hash (similar to `to_hash` method but
32
- includes only attributes mapped to the model and with model attributes names) and copy errors from the model(s)
33
- into a from object.
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 :model
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 :model
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
- ```ruby
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 :model %>
133
- <%= fc.text_field :model %>
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 :model
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 :model
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 on which adds a new it.
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.model = '340 F1'
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
- model: 'M2B',
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.model # => "M2B"
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 and the new one.
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 behaviour is the subject of changes in future releases - only elements with flag `_destroy == true` will be deleted).
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 a primary key.
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 :model, primary_key: true # <- primary key is specified on attribute level
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: :model do # <- primary key is specified on form level
273
- attribute :model
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.model = '340 F1'
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.model = 'M2B'
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
- model: 'M2B',
355
+ code: 'M2B',
306
356
  driver: 'Bruce McLaren',
307
357
  engine: {
308
358
  volume: 3.0
309
359
  }
310
360
  }, {
311
- model: 'M7A',
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].model # => "M2B"
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].model # => "M7A"
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
- # => :model => "340 F1",
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
- # => :model => "M2B",
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
- # => :model => "M7A",
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 `Mappable` mix-in and map form object attributes to one or few models by using `:model` and `:model_attribute` parameters.
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 Mappable
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 Mappable
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 Mappable
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 Mappable
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 :model
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.model` | `model.model` |
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 `hash: true` in order to map a nested form object to a hash as a model.
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 Mappable
542
+ include ModelMapper
492
543
 
493
544
  attribute :name, model_attribute: :team_name
494
545
  attribute :year
495
- attribute :car, hash: true do # nesting only in form object but not in a model
496
- attribute :model
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.model` | `model.car[:model]` |
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 Mappable
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. Save Form Object to Models
595
+ ### 6. Sync Form Object to Models
545
596
 
546
- Use `save_to_models(models)` to save form object attributes to mapped models.
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 Mappable
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.save_to_models(default: default_model, car: car_model)
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 `save_to_models(default: model)` or `save_to_model(model)` to save to single model.
621
+ Use `sync_to_models(default: model)` or `sync_to_model(model)` to sync to single model.
571
622
 
572
- Neither `save_to_models` nor `save_to_model` calls `save` method on the model(s).
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 values to mapped model 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 Mappable
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 :model, primary_key: true # <- primary key is specified on attribute level
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 Mappable
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 :model, primary_key: true # <- primary key is specified on attribute level
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 Mappable
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 Mappable
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 :model, primary_key: true
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
- model: 'M2B',
715
+ code: 'M2B',
665
716
  driver: 'Bruce McLaren'
666
717
  }, {
667
- model: 'M7A',
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
- # => :model => "M2B",
727
+ # => :code => "M2B",
677
728
  # => :driver => "Bruce McLaren"
678
729
  # => }, {
679
- # => :model => "M7A",
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 Mappable
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 :model
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 Mappable
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 :model
835
+ attribute :code
785
836
  attribute :engine_power, model_attribute: 'engine.:power'
786
837
 
787
- validates :model, presence: true
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.save_to_model(model = Team.new)
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.save_to_model(model)
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.model %> (<%= car.engine[:power] %> hp)</li>
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 :model %>
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
- | hash | block* or `:class`** | `false` | `FormObj::Mappable` | 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>` |
881
- | model | - | `:default` | `FormObj::Mappable` | The name of the model to which this attribute is mapped |
882
- | model_attribute | - | `<attribute_name>` | `FormObj::Mappable` | 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. |
883
- | model_class | block* or `:class`** or dot notation for `:model_attribute`*** | `<attribute_name>.classify` | `FormObj::Mappable` | The class (or the name of the class) of the mapped model. |
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 "tree_struct", ">= 1.0.2"
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
@@ -0,0 +1,9 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "activesupport", "~> 3.2.0"
6
+ gem "activemodel", "~> 3.2.0"
7
+ gem "actionpack", "~> 3.2.0"
8
+
9
+ gemspec path: "../"