Join the social network of Tech Nerds, increase skill rank, get work, manage projects...
 
  • Form objects with Reform gem (Rails refactoring)

    • 0
    • 0
    • 0
    • 0
    • 0
    • 0
    • 0
    • 0
    • 1.42k
    Comment on it

    We have been discussing refactoring in our previous posts. Next to our refactoring pattern, we will discuss few gems that can be used to improve our coding style.

     

    In my last post i explained form objects pattern. Form objects are nothing but a replacement of your model object for form. Controllers are meant to receive request and provide result to the views. We can extract the form create/update functionality from controller and abstract it in a form object class.

     

    Benefits of form object pattern are:

    1. It abstract the functionality in a class which is concerned about that particular functionality only as stated by SRP (single responsibility principle).
    2. Extract validations out of models that may not be necessary.
    3. Reduces complexity in saving associated records (multiple model form).


    For this tutorial we will look in to Reform gem which helps to create form objects. It maintains validations for one or multiple models, where a model can be any kind of Ruby object. Add reform gem to gemfile and bundle install.

     

    1. gem "reform"

     

    Lets see an example:
     

    1. class Author < ActiveRecord::Base
    2. has_many :books
    3. end
    4.  
    5.  
    1. class Books < ActiveRecord::Base
    2. belongs_to :author
    3. end
    4.  
    5.  


    We create a form object class that inherits from Reform::Form. We define attributes with property and validations in the class. Reform provides support for nested objects. As we have books association in our Author model. We can define this association in our form class with collection keyword.

     

    1. class AuthorCreateForm < Reform::Form
    2. property :name
    3. validates :name, presence: true
    4.  
    5. collection :books do
    6. property :name
    7. property :description
    8. property :publish_on
    9. validates :name, presence: true
    10. end
    11. end


    Now in our controller we can use this:

    1. class AuthorsController
    2. def new
    3. @form = AuthorCreateForm.new(Author.new)
    4. end
    5.  
    6. def create
    7. @form = AuthorCreateForm.new(Author.new)
    8. if @form.validate(params[:author])
    9. @form.save
    10. else
    11. # handle validation errors.
    12. end
    13. end
    14. end


    Note: The #validate method first updates the values of the form. However, the model object does not save. It then runs validation. The values are saved only after #save or #sync operation are called on form object.


    Passing this form object in our view:

    1. = form_for @form do |f|
    2. = f.input :name
    3.  
    4. = f.fields_for :books do |a|
    5. = a.text_field :name
    6.  
    7. ......

     

    NOTE: The save method on the form object will call #save on the model and nested models.


    Conclusion:

    Isn't that nice, we have extracted the functionality from controller in separate class. Our controllers are now more clean and skinny. Also our controller no longer deals with create/update logic.

    You may use simple ruby class or Reform gem for form objects. Form objects pattern is useful with complex nested model forms. Reform provides us neat ways to define fields and associations.

    For more details and options on reform gem please visit https://github.com/apotonick/reform

 0 Comment(s)

Sign In
                           OR                           
                           OR                           
Register

Sign up using

                           OR                           
Forgot Password
Reset Password
Fill out the form below and reset your password: