No release in over 3 years
Low commit activity in last 3 years
There's a lot of open issues
This is a Rails engine for quickly setting up a flexible publication module.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Development

Runtime

 Project Readme

Fiat Publication

This engine is designed to be used by @fiatinsight developers on Rails projects to speed up development of a custom content system — like a CMS or CRM.

Currently installed on Tekne, Parish.es, and Cleveland Mixer.

Installation

Add this line to your application's Gemfile:

git_source(:github) { |repo| "https://github.com/#{repo}.git" }
gem 'fiat_publication', github: 'fiatinsight/fiat_publication'

For stable release with older applications using this gem prior to November 2019, include tag: 'v1.1.0' in the gem dependency.

Then bundle and run the required migrations directly by typing:

$ rake db:migrate

Create an initializer at config/initializers/fiat_publication.rb to set any required globals for your implementation. You should set any variables that pertain to parts of the gem you'll want to use. Variables give you the chance to pass namespacing into the engine's actions:

FiatPublication.new_message_redirect_path = "message_path"
FiatPublication.new_page_path = "new_admin_page_path"
FiatPublication.view_page_path = "page_path"
FiatPublication.new_page_redirect_path = "page_path"
FiatPublication.new_article_redirect_path = "new_admin_article_path"
FiatPublication.new_content_block_redirect_path = "content_block_path"

Note: This section needs attention to improve the flexibility of the routing, especially with namespaced usage.

Then mount the engine in your routes.rb file:

mount FiatPublication::Engine => "/publication"

You can also mount the engine within a namespace, for example:

namespace :account do
  mount FiatPublication::Engine => "/publication"
  # More routes here...
end

Dependencies

The engine supplies minimally formatted output so that you can influence designs within your main application. It includes dependencies for simple_form, trix-rails, audited, recaptcha, meta-tags, and activestorage-validator. It requires that you've set up Active Storage in your main app. It also requires a Tokenable concern (i.e., generating a token on a create callback) for models to handle tokenization.

It assumes—but doesn't require—that you're using Bootstrap and fiat_ui, as well as fiat_notifications.

Note: A User class is currently required by the mention_users method on the Comment model. This assumes, too, that fiat_notifications is installed, which will eventually be a dependency for this gem.

Content types

This engine supplies a variety of content types that can be invoked in custom configurations to do whatever you need. There are some guiding ideas, though, that'll help to implement the available resources better.

Pages and articles

Pages and articles are basic content types. Pages are intended for more permanent content, and articles are designed to be more transient / ephemeral. You can invoke a new page with:

= link_to send("#{FiatPublication.new_page_path}", publisher_type: nil, publisher_id: nil)

Editing a page requires a little more information. Create a view partial with the following information:

locals = {
  page: FiatPublication::Page.find(params[:id]),
  page_update_url: account_fiat_publication.page_path,
  new_content_block_url: account_fiat_publication.new_content_block_path(publishable_type: 'FiatPublication::Page', publishable_id: params[:id]),
  content_block_path: '/account/publication/content_blocks',
  btn_classes: 'btn btn-success layer-1',
  nested_parent_id: nil, # Identify a parent resource in a nested setup
  publisher_type: nil,
  publisher_id: nil
  }

= render partial: 'fiat_publication/pages/edit', locals: locals

TODO: This section needs considerably more detail around working with engine-supplied page and article forms, writing your own forms, required parameters, routing options, etc.

Content blocks

Content blocks are granular elements used to build pages and articles. By default, the Page and Article classes in this gem use has_many polymorphic associations for content blocks as publishable.

You can also extend a similar relationship to any model(s) in your main app. For example:

class MyPage < ApplicationRecord
  # ...
  has_many :fiat_publication_content_blocks, as: :publishable, class_name: 'FiatPublication::ContentBlock'
end

Content blocks also require a block_type value that can be set to text, image, or buttons. The type of block determines what fields are made available to utilize. Content blocks with images use Active Storage to store and process image files.

Note: More block types will be forthcoming.

Authors

Authors can be created and associated with articles.

TODO: Extend the Author class to accommodate more information, and to associate with other main app classes, like User.

Messages and comments

Messages enable threaded content, similar to email. Adding comments to messages allows you to extend them indefinitely. A message automatically has_many comments; but comments can also be added to other content objects in the engine, or within your main app, using a similar commentable polymorphic relationship.

You only need to supply a subject to create a message.

TODO: This section needs more detail around working with engine-supplied message forms, writing your own forms, including custom fields and labels, etc.

reCaptcha v3

Messages allow you to add reCaptcha v3 to your message form. The create action checks for the verify_recaptcha method and processes it with that if available. The v3 action you include in a new message form must be create_message, for example: = recaptcha_v3(action: 'create_message').

Attachments

Attachments are standalone content objects that use Active Storage to store files.

TODO: This content type is undeveloped. Extrapolating current attachment logic, routing, etc., in conjunction w/ best practice Active Storage use is required. (See this issue.)

Custom fields

Custom fields can be programmed into forms on the main application. Within the engine, they're automatically associated as publishable with messages, articles, and pages. Each of these classes also accepts nested attributes for custom fields, allowing you to include them in custom forms within your main application.

Content labels

Content labels are designed to work like Gmail labels: They're flattened objects that can be associated with individual instances of a class to make taxonomies easier and super flexible. Content labels are automatically associated to pages, articles, and messages in the engine via the ContentLabelAssignment class as assignable.

Navigation

TODO: Navigation items and groups, drawing in engine classes, need to be extrapolated from current use. (See this issue.)

Snoozing

Snoozing is available for messages, by default. You can create a link to snooze something like this:

= link_to "Snooze this", fiat_publication.new_snooze_path(snoozable_type: "FiatPublication::Message", snoozable_id: @message.id, snoozer_type: "User", snoozer_id: current_user.id), remote: true

Set up automatic unsnoozing in your Heroku app using rake unsnooze_things in a cron scheduler.

Publishers

Content can optionally be assigned to a polymorphic publisher as a class from your main app. This can also be omitted, in case there's only one content creator (e.g., in a simple, non-scaled application). You can listen for publisher content on any model(s) you want:

class Organization < ApplicationRecord
  # ...
  has_many :fi_pages, as: :publisher, class_name: 'FiatPublication::Page'
  has_many :fi_articles, as: :publisher, class_name: 'FiatPublication::Article'
end

Routing

New items

Depending on where you mount the engine, routing to its resources will work differently. For example, within an account namespace, a new content block could be created using something like:

link_to "New block", account_fiat_publication.new_content_block_path(publishable_type: "Page", publishable_id: @page.id)

Updating items

In this case, updating content would require passing in the full namespace so that fiat_publication can handle a nested path helper in its forms. That means including the object you want to work with, e.g., setting a content_block variable, as well as the url variable of the path you want to work with:

content_block = FiatPublication::ContentBlock.find(your_content_block_id)
= render partial: 'fiat_publication/content_blocks/form', locals: { content_block: content_block, url: account_fiat_publication.content_block_path(content_block) }

Creating a new entry would take similar arguments:

= render partial: 'fiat_publication/messages/new', locals: { message: FiatPublication::Message.new, url: account_fiat_publication.messages_path }
# Note the use of content_blocks_path and not new_content_block_path

Redirecting

In addition to the default redirect variables in your initializer, you can alternatively pass a redirect_path variable within the url parameter of a form. This processes using the send() function, so you can write it like this:

= render partial: 'fiat_publication/messages/new', locals: { message: FiatPublication::Message.new, url: account_fiat_publication.messages_path(redirect_path: root_path) }

Or like this:

= render partial: 'fiat_publication/messages/new', locals: { message: FiatPublication::Message.new, url: account_fiat_publication.messages_path(redirect_path: "'account_fiat_publication.message_path', @message") }

Notices

You can pass a notice variable in the url parameter, too:

= render partial: 'fiat_publication/messages/new', locals: { message: FiatPublication::Message.new, url: account_fiat_publication.messages_path(redirect_path: root_path, notice: "That was awesome. Well done!") }

Displaying items

Displaying content just requires that you use the typical associations. For example, if you wanted to display the default partial for a content block provided by fiat_publication you could put:

@page.fiat_publication_content_blocks.each do |i|
  = render partial: 'fiat_publication/content_blocks/show', locals: { content_block: i, my_classes: 'custom_classes_here' }
end

Extending classes

You can extend classes by using decorators at /app/decorators/**/*_decorator*.rb

For example, a file called /app/decorators/models/fiat_publication/message_decorator.rb might include:

FiatPublication::Message.class_eval do
  def new_method
    # Some code
  end
end

Note: Be sure to restart your application when introducing decorators

Customization

You can use the gem resources directly, or wrap them into custom namespaces, views, and controller logic within your main app. For example, after mounting it within a namespace called account, you could create a series of controllers under your AccountController to handle provided resources, e.g., Account::PagesController or Account::ArticlesController. You can do this for some resources or all of them. Make sure to create routes for each resource type you want to handle:

namespace :account do
  mount FiatPublication::Engine => "/publication"
  resources :pages
  resources :articles
end

You'll also need to add variables to your initializer file to tell the gem how to handle redirects for different content types. For example, with the account namespace, you might set:

FiatPublication.new_content_block_redirect_path = "account_content_block_path"

The gem will provide arguments for your paths based on the resource type. But it can't pick up the namespace conventions you set in your main app without a little help.

Development

To build this gem for the first time, run gem build fiat_publication.gemspec from the project folder.

To install this gem onto your local machine, run bundle exec rake install. To release a new version, update the version number in version.rb, and then run bundle exec rake release, which will create a git tag for the version, push git commits and tags, and push the .gem file to rubygems.org.

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/fiatinsight/fiat_publication.

License

The gem is available as open source under the terms of the MIT License.