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.
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
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.
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 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
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>
This is a somewhat contrived example; you’d almost certainly be better off mapping the data much earlier. ↩
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) ↩