require 'hanami/helpers/form_helper/form_builder' module Hanami module Helpers # Form builder # # By including Hanami::Helpers::FormHelper it will inject one public method: form_for. # This is a HTML5 form builder. # # To understand the general HTML5 builder syntax of this framework, please # consider to have a look at Hanami::Helpers::HtmlHelper documentation. # # This builder is independent from any template engine. # This was hard to achieve without a compromise: the form helper should be # used in one output block in a template or as a method in a view (see the examples below). # # Features: # # * Support for complex markup without the need of concatenation # * Auto closing HTML5 tags # * Support for view local variables # * Method override support (PUT/PATCH/DELETE HTTP verbs aren't understood by browsers) # * Automatic generation of HTML attributes for inputs: id, name, value # * Allow to override HTML attributes # * Extract values from request params and fill value attributes # * Automatic selection of current value for radio button and select inputs # * Infinite nested fields # # Supported tags and inputs: # # * check_box # * color_field # * date_field # * datetime_field # * datetime_local_field # * email_field # * fields_for # * file_field # * form_for # * hidden_field # * label # * number_field # * password_field # * radio_button # * select # * submit # * text_area # * text_field # # @since 0.2.0 # # @see Hanami::Helpers::FormHelper#form_for # @see Hanami::Helpers::HtmlHelper # # @example One output block (template) # <%= # form_for :book, routes.books_path do # text_field :title # # submit 'Create' # end # %> # # @example Method (view) # require 'hanami/helpers' # # class MyView # include Hanami::Helpers::FormHelper # # def my_form # form_for :book, routes.books_path do # text_field :title # end # end # end # # # <%= my_form %> module FormHelper # Default HTTP method for form # # @since 0.2.0 # @api private DEFAULT_METHOD = 'POST'.freeze # Default charset # # @since 0.2.0 # @api private DEFAULT_CHARSET = 'utf-8'.freeze # CSRF Token session key # # This key is shared with hanamirb, hanami-controller. # # @since 0.2.0 # @api private CSRF_TOKEN = :_csrf_token # Form object # # @since 0.2.0 class Form # @return [Symbol] the form name # # @since 0.2.0 # @api private attr_reader :name # @return [String] the form action # # @since 0.2.0 # @api private attr_reader :url # @return [::Hash] the form values # # @since 0.2.0 # @api private attr_reader :values # Initialize a form # # It accepts a set of values that are used in combination with request # params to autofill value attributes for fields. # # The keys of this Hash, MUST correspond to the structure of the (nested) # fields of the form. # # For a given input where the name is `book[title]`, Hanami will # look for `:book` key in values. # # If the current params have the same key, it will be PREFERRED over the # given values. # # For instance, if params.get('book.title') equals to # "TDD" while values[:book].title returns # "No test", the first will win. # # @param name [Symbol] the name of the form # @param url [String] the action of the form # @param values [Hash,NilClass] a Hash of values to be used to autofill # value attributes for fields # @param attributes [Hash,NilClass] a Hash of attributes to pass to the # form tag # # @since 0.2.0 # # @example Pass A Value # # Given the following view # # module Web::Views::Deliveries # class Edit # include Web::View # # def form # Form.new(:delivery, routes.delivery_path(id: delivery.id), # {delivery: delivery, customer: customer}, # {method: :patch}) # end # end # end # # # And the corresponding template: # # <%= # form_for form do # date_field :delivered_on # # fields_for :customer do # text_field :name # # fields_for :address do # # ... # text_field :city # end # end # # submit 'Update' # end # %> # # # #
# # # # # Value taken from delivery.delivered_on # # # # Value taken from customer.name # # # # Value taken from customer.address.city # # # #
def initialize(name, url, values = {}, attributes = {}) @name = name @url = url @values = values @attributes = attributes || {} end # Return the method specified by the given attributes or fall back to # the default value # # @return [String] the method for the action # # @since 0.2.0 # @api private # # @see Hanami::Helpers::FormHelper::DEFAULT_METHOD def verb @attributes.fetch(:method, DEFAULT_METHOD) end end # Instantiate a HTML5 form builder # # @overload form_for(name, url, options, &blk) # Use inline values # @param name [Symbol] the toplevel name of the form, it's used to generate # input names, ids, and to lookup params to fill values. # @param url [String] the form action URL # @param options [Hash] HTML attributes to pass to the form tag and form values # @option options [Hash] :values An optional payload of objects to pass # @param blk [Proc] A block that describes the contents of the form # # @overload form_for(form, attributes = {}, &blk) # Use Form # @param form [Hanami::Helpers::FormHelper::Form] a form object # @param attributes [Hash] HTML attributes to pass to the form tag and form values # @param blk [Proc] A block that describes the contents of the form # # @return [Hanami::Helpers::FormHelper::FormBuilder] the form builder # # @since 0.2.0 # # @see Hanami::Helpers::FormHelper # @see Hanami::Helpers::FormHelper::Form # @see Hanami::Helpers::FormHelper::FormBuilder # # @example Inline Values In Template # <%= # form_for :book, routes.books_path, class: 'form-horizontal' do # div do # label :title # text_field :title, class: 'form-control' # end # # submit 'Create' # end # %> # # # #
# #
# # #
# # #
# # # # @example Use In A View # # module Web::Views::Books # class New # # def form # form_for :book, routes.books_path, class: 'form-horizontal' do # div do # label :title # text_field :title, class: 'form-control' # end # # submit 'Create' # end # end # end # # # <%= form %> # # # #
# #
# # #
# # #
# # @example Share Code Between Views # # # Given the following views to create and update a resource # module Web::Views::Books # class New # include Web::View # # def form # Form.new(:book, routes.books_path) # end # # def submit_label # 'Create' # end # end # # class Edit # include Web::View # # def form # Form.new(:book, routes.book_path(id: book.id), # {book: book}, {method: :patch}) # end # # def submit_label # 'Update' # end # end # end # # # The respective templates can be identical: # # ## books/new.html.erb # <%= render partial: 'books/form' %> # # ## books/edit.html.erb # <%= render partial: 'books/form' %> # # # While the partial can have the following markup: # # ## books/_form.html.erb # <%= # form_for form, class: 'form-horizontal' do # div do # label :title # text_field :title, class: 'form-control' # end # # submit submit_label # end # %> # # # #
# #
# # #
# # #
# # @example Method override # <%= # form_for :book, routes.book_path(id: book.id), method: :put do # text_field :title # # submit 'Update' # end # %> # # # #
# # # # # #
# # @example Nested fields # <%= # form_for :delivery, routes.deliveries_path do # text_field :customer_name # # fields_for :address do # text_field :city # end # # submit 'Create' # end # %> # # # #
# # # # # #
def form_for(name, url = nil, options = {}, &blk) form = if name.is_a?(Form) options = url || {} name else Form.new(name, url, options.delete(:values)) end attributes = { action: form.url, method: form.verb, 'accept-charset': DEFAULT_CHARSET, id: "#{form.name}-form" }.merge(options) FormBuilder.new(form, attributes, self, &blk) end # Returns CSRF Protection Token stored in session. # # It returns nil if sessions aren't enabled or the value is missing. # # @return [String,NilClass] token, if present # # @since 0.2.0 def csrf_token if defined?(session) session[CSRF_TOKEN] elsif defined?(locals) && locals[:session] locals[:session][CSRF_TOKEN] end end end end end