Customizing Blacklight Tutorial Series (v8.0.0.alpha)

Configuring Metadata Fields

The metadata fields that are displayed in the index (search results) and show (record view) can be configured using the add_index_field and add_show_field configuration methods in the Blacklight config.

configure_blacklight do |config|
  # Index / Search Results
  config.add_index_field 'author_tsim', label: 'Author'

  # Show / Record View
  config.add_show_field 'author_tsim', label: 'Author'
end

The configuration patterns outlined on General Configuration Patterns apply to these fields so you can use any of those to customize the display of these fields.

Labeling via Internationalization (i18n)

All the text in Blacklight’s UI are configured via rails’ i18n files. While you can apply a string label directly in the Blackklight config, you can also label these in your i18n config files (e.g. in config/locales/blacklight.en.yml).

If you wanted to change the label of all author_tsim fields to “Creator” you can do that by adding the label to your locale file.

en:
  blacklight:
    search:
      fields:
        author_tsim: Creator

The context of where the fields are being rendered in the application will be passed though as part of the i18n key as well, so you can have different labels under different view contexts (e.g. index vs show) or have a default that is overridden for a particular view context.

en:
  blacklight:
    search:
      fields:
        title_tsim: Title
        index:
          author_tsim: Author
        show:
          author_tsim: Creator
          title_tsim: Work

Formatting Options

There are a few configuration options that can help you format the data using Blacklight out-of-the-box rendering pipeline.

One common customization is to update the way that Blacklight joins multiple values by default. Blacklight’s rendering pipeline uses rails’ to_sentence helper (with its default options) to join multiple solr field values together, and allows you to pass in alternate options using the separator_options configuration option. For instance, if we wanted to separate all values by line breaks instead of the ,s and and we can do that with the following configuration (the record with ID 92117465 is a good example of this in action).

configure_blacklight do |config|
  # This field configuration would need to be added as it is not in the generated controller
  config.add_show_field 'author_addl_tsim', label: 'Additional authors', separator_options: {
    two_words_connector: '<br />',
    words_connector: '<br />',
    last_word_connector: '<br />'
  }
end

It’s also possible to link these values using the built-in link_to_facet configuration option. Two of the show fields that it might make sense to do this with (and are indexed appropriately for faceting) are the format and language fields on the show view.

- config.add_show_field 'format', label: 'Format'
- config.add_show_field 'language_ssim', label: 'Language'
+ config.add_show_field 'format', label: 'Format', link_to_facet: true
+ config.add_show_field 'language_ssim', label: 'Language', link_to_facet: true

These two fields should now be linked to a faceted search result for their respective fields.

Values

The field values can also be mapped using field configuration. There are two easy ways to transform values from the solr document into values to display to the end user; you can use a model method/accessor for data-based transformations, or a controller-level method for other transformations.

Accessors

Accessors are handy when the transformation you want to make is based entirely on the data from the field (or document). One thing you might do is map names given as “last name, first name” into “first name last name” 1. The most basic accessor configuration is using the explicit accessor configuration:

configure_blacklight do |config|
  # Index / Search Results
  config.add_index_field 'author_tsim', label: 'Author', accessor: :author_name
end

This will call the accessor method (author_name) on the SolrDocument instance for each result. In app/models/solr_document.rb, we need to implement that method:

class SolrDocument
  def author_name
    return unless has? 'author_tsim'

    # Don't really do this; find a way to do this wherever you have enough information
    # to do it without as many errors.
    Array(fetch('author_tsim')).map do |value|
      a, b, c = value.split(', ', 3)

      if a && b && c
        # the value had all 3 parts
        "#{b} #{a}, #{c}"
      elsif a && !b&.match?(/^\d/)
        # the value has just two parts, and the second part was probably not a year
        "#{b} #{a}"
      else
        value
      end
    end
  end
end

Alternatively, you can use the values parameter to provide document values 2. This is a good choice if the value varies based on the request (e.g. the values are internationalized using the user’s profile, or an administrator sees additional data, etc):

configure_blacklight do |config|
  # Index / Search Results
  config.add_index_field 'JSON', label: 'Solr Document', values: ->(field_config, document, _) {
    "<pre>#{document.as_json}</pre>".html_safe
  }
end

Components

The field values can also be rendered using a custom viewcomponent. This is a good choice if you want to render a field value in a more complex way than the default rendering pipeline. With a custom view component, you can render the field value in any way you want.

config.add_show_field 'JSON', label: 'Solr Document', component: DetailsComponent

In app/components/details_component.rb:

class DetailsComponent < Blacklight::MetadataFieldComponent
end

In app/components/details_component.html.erb:

<div>
  <details>
    <summary><%= @field.label %></summary>

    <pre><%= @field.values.first %></pre>
  </details>
</div>
  1. This is a somewhat contrived example; you’d almost certainly be better off mapping the data much earlier. 

  2. We’re using an inline lambda here for expediency; you could (or even should) structure your code for maintainability using any number of approaches (including extracting it as a constant; extracting a method and using the method method to lambda-fy it; extracting a service class that responds to #call; etc)