# frozen_string_literal: true require_relative "page_meta_for/version" require 'active_support' require 'action_controller' module PageMeta extend ::ActiveSupport::Concern included do initialize_meta helper_method :page_meta_for end module ClassMethods def initialize_meta before_action do @page_meta_for = {} end end # Controller level method to define the part of the meta tags # # class ApplicationController < ActionController::Base # meta_for(:title, :suffix, 'ApplicationName') # meta_for(:title, :prefix) { action_name } # meta_for(:description) { 'Lorem ipsum set' } # end # # class WelcomeController < ApplicationController # meta_for(:title) { 'Landing Page' } # meta_for(:description) { 'Lorem ipsum set2' } # end # # Then views and layouts are able to call `page_meta_for`: # # <%= page_meta_for(:title, join_string: ' * ') %> # # # It renders the following: # # Index * Landing Page * ApplicationName # # # If last argument is Hash, a appropriate method will be called # # class ApplicationController < ActionController::Base # meta_for(:title, :generate_title) # # private # # def generate_title # 'Application Name' # end # end # def meta_for(*keys, &block) content_proc = if block_given? block else c = keys.pop c.is_a?(Hash) ? proc { send(c) } : proc { c } end keys = Array.wrap(keys).flatten key = keys.pop scope = keys.pop || :value before_action do @page_meta_for ||= {} @page_meta_for[scope] ||= {} @page_meta_for[scope][key] = instance_exec(&content_proc).presence || nil end end end def page_meta_for(key, scopes: [:prefix, :value, :suffix], join_string: " - ") @page_meta_for ||= {} scopes.map { |s| @page_meta_for.fetch(s, {}).fetch(key, nil) } .select(&:present?) .join(join_string) .presence end end ActionController::Base.include(PageMeta)