# Effective Bootstrap Everything your Ruby on Rails 5.1+ application needs to get working with [Twitter Bootstrap 4](https://getbootstrap.com/). - Bootstrap4 component view helpers. - SVG icons based on [Inline SVG](https://github.com/jamesmartin/inline_svg), with [Feather Icons](https://feathericons.com) and [FontAwesome](https://fontawesome.com) svg icons to replace the old glyphicons. - An html-exact form builder that builds on top of Rails' new `form_with` with numerous custom form inputs. ## Getting Started ```ruby gem 'effective_bootstrap' ``` Run the bundle command to install it: ```console bundle install ``` Install the configuration file: ```console rails generate effective_bootstrap:install ``` The generator will install an initializer which describes all configuration options. Add the following to your `application.js`: ```ruby //= require jquery3 //= require popper //= require bootstrap //= require effective_bootstrap # The date picker form inputs use momentjs locales. To add a locale: //= require moment/locale/es //= require moment/locale/nl ``` And to your `application.scss`: ```sass @import 'bootstrap' @import 'effective_bootstrap' ``` ## View Helpers All these examples are in [haml](https://github.com/haml/haml). ### Collapse https://getbootstrap.com/docs/4.0/components/collapse/ ```haml = collapse('Click to collapse/expand') do %p You have revealed me! ``` ### Dropdown https://getbootstrap.com/docs/4.0/components/dropdowns/ ```haml = dropdown do = dropdown_link_to 'Something', root_path = dropdown_divider = dropdown_link_to 'Another', root_path ``` Options include: `dropdown(variation: :dropup|:dropleft|:dropright, split: true|false, right: true|false, btn: 'btn-secondary')` ### ListGroup https://getbootstrap.com/docs/4.0/components/list-group/ ```haml = list_group do = list_group_link_to 'Something', root_path ``` `list_group_link_to` will automatically insert the the `.active` class based on the request path. ### Navbar https://getbootstrap.com/docs/4.0/components/navbar/ ```haml %nav.navbar.navbar-expand-lg.navbar-light.bg-light %a.navbar-brand{href: '/'} Home %button.navbar-toggler{type: 'button', data: {toggle: 'collapse', target: '#navContent', 'aria-controls': 'navContent', 'aria-label': 'Toggle navigation'}} %span.navbar-toggler-icon #navContent.collapse.navbar-collapse %ul.navbar-nav.mr-auto = nav_link_to 'About', '/about' = nav_link_to 'Contact', '/conact' %ul.navbar-nav - if current_user.present? = nav_dropdown('Account', right: true) do = nav_link_to 'Settings', user_settings_path - if can?(:access, :admin) = nav_divider = nav_link_to 'Site Admin', '/admin' = nav_divider = nav_link_to 'Sign Out', destroy_user_session_path, method: :delete - else = nav_link_to 'Sign In', new_user_session_path ``` `nav_link_to` will automatically insert the `.active` class based on the request path. ### Pagination https://getbootstrap.com/docs/4.0/components/pagination/ Builds a pagination based on the given collection, current url and params[:page]. The collection must be an ActiveRecord relation. ```haml = paginate(@posts, per_page: 10) ``` Add this to your model: ```ruby scope :paginate, -> (page: nil, per_page:) { page = (page || 1).to_i offset = [(page - 1), 0].max * per_page limit(per_page).offset(offset) } ``` Add this to your controller: ```ruby def index @posts = Post.all.paginate(page: params[:page], per_page: 10) end ``` Add this to your view: ```haml %nav= paginate(@posts, per_page: 10) ``` or ```haml %nav.d-flex.justify-content-center= paginate(@posts, per_page: 10) ``` ### Tabs https://getbootstrap.com/docs/4.0/components/navs/#tabs ```haml = tabs do = tab 'Demographics' do %p Demographics tab = tab 'Orders' do %p Orders tab - if resource.logs.present? = tab 'Logs' do %p Logs tab ``` ## Icon Helpers Unfortunately, Bootstrap 4 dropped support for glyphicons, so we use a combination of [Inline SVG](https://github.com/jamesmartin/inline_svg), with [Feather Icons](https://feathericons.com) and [FontAwesome](https://fontawesome.com) .svg images (no webfonts) to get back this functionality, even better than it was before. ```haml = icon('ok') # ``` ```haml = icon_to('ok', root_path) # ``` A full list of icons can be found here: [All effective_bootstrap icons](https://github.com/code-and-effect/effective_bootstrap/tree/master/app/assets/images/icons) To overwrite or add an icon, just drop the `.svg` file into your application's `app/assets/images/icons/` directory. There are also a few helpers for commonly used icons, they all take the form of `x_icon_to(new_thing_path)`: - `new_icon_to` - `show_icon_to` - `edit_icon_to` - `destroy_icon_to` - `settings_icon_to` - `ok_icon_to` - `approve_icon_to` - `remove_icon_to` ## Form Builder Rails 5.1 has introduced a new `form_with` syntax, and soft-deprecated `form_tag` and `form_for`. This gem includes a [Bootstrap4 Forms](https://getbootstrap.com/docs/4.0/components/forms/) html-exact form builder built on top of `form_with`. The goal of this form builder is to output beautiful forms while matching the rails form syntax -- you should be able to change an existing `form_with` form to `effective_form_with` with no other changes. Of course, just the regular form inputs are boring, and this gem extends numerous jQuery/Javascript libraries to level up some inputs. This is an opinionated Bootstrap4 form builder. ## effective_form_with Matches the Rails `form_with` tag syntax, with all its `:model`, `:scope`, `:url`, `:method`, etc. As well, you can specify `layout: :vertical`, `layout: :horizontal`, or `layout: :inline` as per the different Bootstrap form layouts. ```haml = effective_form_with(model: @user, layout: :horizontal) do |f| = f.text_field :name = f.submit ``` The default is `layout: :vertical`. If you would like each form and its fields to have unique ids, use `unique_ids: true`. All standard form fields have been implemented as per [Rails 5.1 FormHelper](http://api.rubyonrails.org/v5.1/classes/ActionView/Helpers/FormHelper.html) When working as a `remote: true` form, you can also pass `flash_success: true|false` and `flash_error: true|false` to control the flash behaviour. By default, the errors will be displayed, and the success will be hidden. ### Options There are three sets of options hashes that you can pass into any form input: - `wrapper: { class: 'something' }` are applied to the wrapping div tag. - `input_html: { class: 'something' }` are applied to the input, select or textarea tag itself. - `input_js: { key: value }` are passed to any custom form input will be used to initialize the Javascript library. For example: ```ruby = effective_form_with(model: @user) do |f| = f.date_field :updated_at, input_js: { useCurrent: 'day', showTodayButton: true } ``` will result in the following call to the Javascript library: ```javascript $('input').datetimepicker(useCurrent: 'day', showTodayButton: true); ``` Any options passed in this way will be used to initialize the underlying javascript libraries. ## Basic form inputs The following form inputs are supported, but don't have any kind of custom JavaScript ```haml = f.check_box = f.email_field = f.error_field = f.number_field = f.password_field = f.static_field = f.text_area = f.text_field = f.url_field ``` ## Custom date_field, datetime_field, time_field These custom form inputs are all based on the following awesome project: Bootstrap 3 Datepicker (https://github.com/Eonasdan/bootstrap-datetimepicker) ```haml = f.date_field :updated_at = f.datetime_field :updated_at = f.time_field :updated_at ``` ### Options The default options used to initialize this form input are as follows: ```ruby am_pm: true, input_js: { showTodayButton: false, showClear: false, useCurrent: 'hour' } ``` For a full list of options, please refer to: http://eonasdan.github.io/bootstrap-datetimepicker/Options/ ### Set Date Use the following JavaScript to set the date: ```javascript $('#start_at').data('DateTimePicker').date('2016-05-08') ``` ### Disabled Dates Provide a String, Date, or Range to set the disabled dates. ```ruby input_js: { disabledDates: '2020-01-01' } input_js: { disabledDates: Time.zone.now } input_js: { disabledDates: Time.zone.now.beginning_of_month..Time.zone.now.end_of_month } input_js: { disabledDates: [Time.zone.now, Time.zone.now + 1.day] } ``` ### Linked Dates By default, when two matching date inputs named `start_*` and `end_*` are present on the same form, they will become linked. The end date selector will have its date <= start_date disabled. To disable this behaviour, call with `date_linked: false`. ```ruby = f.date_field :end_at, date_linked: false ``` ### Events The date picker library doesn't trigger a regular `change`. Instead you must watch for the `dp.change` event. More info is available here: http://eonasdan.github.io/bootstrap-datetimepicker/Events/ ## Custom editor A drop in ready rich text editor based on https://quilljs.com/ To use the editor, you must make additional javascript and stylesheet includes: In your application.js ``` //= require effective_bootstrap //= require effective_bootstrap_editor ``` In your application.scss ``` @import 'effective_bootstrap'; @import 'effective_bootstrap_editor'; ``` And then in any form, instead of a text area: ``` = f.editor :body ``` ## Custom has_many This custom form input was inspired by [cocoon](https://github.com/nathanvda/cocoon) but works with more magic. This nested form builder allows has_many resources to be created, updated, destroyed and reordered. Just add `has_many` and `accepts_nested_attributes_for` like normal and then use it in the form: ```ruby class Author < ApplicationRecord has_many :books accepts_nested_attributes_for :books, allow_destroy: true end ``` and ```haml = effective_form_with(model: author) do |f| = f.text_field :name = f.has_many :books do |fb| = fb.text_field :title = fb.date_field :published_at ``` If `Book` has an integer `position` field, there will be reorder buttons to drag & drop reorder the items. If `allow_destroy: true` there will be remove buttons. You can customize the has many behaviour by passing the following: ```haml = f.has_many :books, add: true, remove: true, reorder: true ``` or add an html class: ```haml = f.has_many :books, class: 'tight' ``` ## Custom percent_field This custom form input uses no 3rd party jQuery plugins. It displays a percentage formatted value `100` or `12.500` but posts the "percentage as integer" value of `100000` or `12500` to the server. It's like the price field, but 3 digits instead of 2. ```haml = f.percent_field :percent ``` ## Custom price_field This custom form input uses no 3rd party jQuery plugins. It displays a currency formatted value `100.00` but posts the "price as integer" value of `10000` to the server. Think about this value as "the number of cents". ```haml = f.price_field :price ``` This gem also includes a rails view helper `price_to_currency` that takes a value like `10000` and displays it as `$100.00` ## Custom select This custom form input is based on the following awesome project: Select2 (https://select2.github.io/) ### Usage As a Rails Form Helper input: ```ruby = f.select :category, 10.times.map { |x| "Category #{x}"} = f.select :categories, 10.times.map { |x| "Category #{x}"}, multiple: true = f.select :categories, 10.times.map { |x| "Category #{x}"}, tags: true = f.select :categories, {'Active': [['Post A', 1], ['Post B', 2]], 'Past': [['Post C', 3], ['Post D', 4]]}, grouped: true ``` ### Modes The standard mode is a replacement for the default single select box. Passing `multiple: true` will allow multiple selections to be made. Passing `freeform: true` will allow a single selection and new ones to be created. Passing `multiple: true, tags: true` will allow multiple selections to be made, and new value options to be created. This will allow you to both select existing tags and create new tags in the same form control. Passing `grouped: true` will enable optgroup support. When in this mode, the collection should be a Hash of ActiveRecord Relations or Array of Arrays ```ruby {'Active' => Post.active, 'Past' => Post.past} {'Active' => [['Post A', 1], ['Post B', 2]], 'Past' => [['Post C', 3], ['Post D', 4]]} ``` Passing `polymorphic: true` will enable polymorphic support. In this mode, an additional 2 hidden input fields are created alongside the select field. So calling ```ruby = f.input :primary_contact, User.all.to_a + Member.all.to_a, polymorphic: true ``` will internally translate the collection into: ```ruby [['User 1', 'User_1'], ['User 2', 'User_2'], ['Member 100', 'Member_100']] ``` and instead of posting to the server with the parameter `:primary_contact`, it will instead post `{primary_contact_id: 2, primary_contact_type: 'User'}`. Using both `polymorphic: true` and `grouped: true` is recommended. In this case the expected collection is as follows: ```ruby = f.input :primary_contact, {'Users': User.all, 'Members': Member.all}, polymorphic: true, grouped: true ``` ### Options The default options used to initialize this form input are as follows: ```ruby { :theme => 'bootstrap', :minimumResultsForSearch => 6, :tokenSeparators => [',', ' '], :width => 'style', :placeholder => 'Please choose', :allowClear => !(options[:multiple]) # Only display the Clear 'x' on a single selection box } ``` ### Interesting Available Options To limit the number of items that can be selected in a multiple select box: ```ruby maximumSelectionLength: 2 ``` To hide the search box entirely: ```ruby minimumResultsForSearch: 'Infinity' ``` For a full list of options, please refer to: https://select2.github.io/options.html The following `input_js: options` are not part of the standard select2 API, and are custom `effective_select` functionality only: To add a css class to the select2 container or dropdown: ```ruby containerClass: 'custom-container-class' dropdownClass: 'custom-dropdown-class' ``` to display rich html for the option value: ```ruby f.select :user, user_tag_collection(User.all), template: :html def user_tag_collection(users) users.map do |user| [ user.to_s, user.to_param, { 'data-html': content_tag(:span, user.to_s, class: 'user-choice') } ] end end ``` ### Additional Call with `single_selected: true` to ensure only the first selected option tag will be `