Source: src/js/views/filters/FilterEditorView.js

define([
  "jquery",
  "underscore",
  "backbone",
  "models/filters/Filter",
  "models/filters/ChoiceFilter",
  "models/filters/DateFilter",
  "models/filters/ToggleFilter",
  "collections/queryFields/QueryFields",
  "views/searchSelect/QueryFieldSelectView",
  "views/filters/FilterView",
  "views/filters/ChoiceFilterView",
  "views/filters/DateFilterView",
  "views/filters/ToggleFilterView",
  "text!templates/filters/filterEditor.html",
], function (
  $,
  _,
  Backbone,
  Filter,
  ChoiceFilter,
  DateFilter,
  ToggleFilter,
  QueryFields,
  QueryFieldSelect,
  FilterView,
  ChoiceFilterView,
  DateFilterView,
  ToggleFilterView,
  Template,
) {
  "use strict";

  /**
   * @class FilterEditorView
   * @classdesc Creates a view of an editor for a custom search filter
   * @classcategory Views/Filters
   * @screenshot views/filters/FilterEditorView.png
   * @since 2.17.0
   * @name FilterEditorView
   * @extends Backbone.View
   * @constructor
   */
  var FilterEditorView = Backbone.View.extend(
    /** @lends FilterEditorView.prototype */ {
      /**
         * A Filter model to be rendered and edited in this view. The Filter model must be
         * part of a Filters collection.
         //  TODO: Add support for boolean and number filters
         * @type {Filter|ChoiceFilter|DateFilter|ToggleFilter}
         */
      model: null,

      /**
       * If rendering an editor for a brand new Filter model, provide the Filters
       * collection instead of the Filter model. A new model will be created and, if the
       * user clicks save, it will be added to this Filters collection.
       * @type {Filters}
       */
      collection: null,

      /**
       * A reference to the PortalEditorView
       * @type {PortalEditorView}
       */
      editorView: undefined,

      /**
       * Set to true if rendering an editor for a brand new Filter model that is not yet
       * part of a Filters collection. If isNew is set to true, then the view requires a
       * Filters model set to the view's collection property. A model will be created.
       */
      isNew: false,

      /**
       * The HTML classes to use for this view's element
       * @type {string}
       */
      className: "filter-editor",

      /**
       * References to the template for this view. HTML files are converted to
       * Underscore.js templates
       * @type {Underscore.Template}
       */
      template: _.template(Template),

      /**
       * The classes to use for various elements in this view
       * @type {Object}
       * @property {string} fieldsContainer - the element in the template that
       * will contain the input where a user can select metadata fields for the custom
       * search filter.
       * @property {string} editButton - The button a user clicks to start
       * editing a search filter
       * @property {string} cancelButton - the element in the template that a
       * user clicks to undo any changes made to the filter and close the editing modal.
       * @property {string} saveButton - the element in the template that a user
       * clicks to add their filter changes to the parent Filters collection and close
       * the editing modal.
       * @property {string} deleteButton - the element in the template that a
       * user clicks to remove the Filter model from the Filters collection
       * @property {string} uiBuilderChoicesContainer - The container for the
       * uiBuilderChoices and the associated instruction text
       * @property {string} uiBuilderChoices - The container for each "button" a
       * user can click to switch the filter type
       * @property {string} uiBuilderChoice - The element that acts like a
       * button that switches the filter type
       * @property {string} uiBuilderChoiceActive - The class to add to a
       * uiBuilderChoice buttons when that option is active/selected
       * @property {string} uiBuilderLabel - The label that goes along with the
       * uiBuilderChoice element
       * @property {string} uiBuilderContainer - The element that will be turned
       * into a carousel that switches between each UI Builder view when a user switches
       * the filter type
       * @property {string} modalInstructions - The class to add to the
       * instruction text in the editing modal window
       */
      classes: {
        fieldsContainer: "fields-container",
        editButton: "edit-button",
        cancelButton: "cancel-button",
        saveButton: "save-button",
        deleteButton: "delete-button",
        uiBuilderChoicesContainer: "ui-builder-choices-container",
        uiBuilderChoices: "ui-builder-choices",
        uiBuilderChoice: "ui-builder-choice",
        uiBuilderChoiceActive: "selected",
        uiBuilderLabel: "ui-builder-choice-label",
        uiBuilderContainer: "ui-builder-container",
        modalInstructions: "modal-instructions",
      },

      /**
       * Strings to use to display various messages to the user in this view
       * @property {string} editButton - The text to show in the button a user clicks to
       * open the editing modal window.
       * @property {string} addFilterButton - The text to show in the button a user
       * clicks to add a new search filter and open an editing modal window.
       * @property {string} step1 - The instructions placed just before the fields input
       * @property {string} step2 - The instructions placed after the fields input and
       * before the uiBuilder select
       * @property {string} filterNotAllowed - The message to show when a filter type
       * doesn't work with the selected metadata fields
       * @property {string} saveButton - Text for the button at the bottom of the
       * editing modal that adds the filter model changes to the parent Filters
       * collection and closes the modal
       * @property {string} cancelButton - Text for the button at the bottom of the
       * editing modal that closes the modal window without making any changes.
       * @property {string} deleteButton - Text for the button at the bottom of the
       * editing modal that removes the Filter model from the Filters collection.
       * @property {string} validationError - The message to show at the top of the
       * modal when there is at least one validation error.
       * @property {string} noFilterOption - The message to show when there is no UI
       * available for the selected field or combination of fields.
       */
      text: {
        editButton: "EDIT",
        addFilterButton: "Add a search filter",
        step1: "Let people filter your data by",
        step2: "...using the following interface",
        filterNotAllowed:
          "This interface doesn't work with the metadata fields you" +
          " selected. Change the 'filter data by' option to use this interface.",
        saveButton: "Use these filter settings",
        cancelButton: "Cancel",
        deleteButton: "Remove filter",
        validationError:
          "Please provide the content flagged below before saving this " +
          "search filter.",
        noFilterOption:
          "There are currently no filter options available to support " +
          "this field, or this combination of fields. Change the 'filter data by' " +
          "option to select an interface.",
      },

      /**
       * A function that returns a Backbone events object
       * @return {object} A Backbone events object - an object with the events this view
       * will listen to and the associated function to call.
       */
      events: function () {
        var events = {};
        events["click ." + this.classes.uiBuilderChoice] =
          "handleFilterIconClick";
        return events;
      },

      /**
       * A list of query fields names to exclude from the list of options in the
       * QueryFieldSelectView
       * @type {string[]}
       */
      excludeFields: MetacatUI.appModel.get("collectionQueryExcludeFields"),

      /**
       * An additional field object contains the properties for an additional query
       * field to add to the QueryFieldSelectView that are required to render it
       * correctly. An additional query field is one that does not actually exist in the
       * query service index.
       *
       * @typedef {Object} AdditionalField
       *
       * @property {string} name - A unique ID to represent this field. It must not
       * match the name of any other query fields.
       * @property {string[]} fields - The list of real query fields that this
       * abstracted field will represent. It must exactly match the names of the query
       * fields that actually exist.
       * @property {string} label - A user-facing label to display.
       * @property {string} description - A description for this field.
       * @property {string} category - The name of the category under which to place
       * this field. It must match one of the category names for an existing query
       * field.
       */

      /**
       * A list of additional fields which are not retrieved from the query service
       * index, but which should be added to the list of options in the
       * QueryFieldSelectView. This can be used to add abstracted fields which are a
       * combination of multiple query fields, or to add a duplicate field that has a
       * different label.
       *
       * @type {AdditionalField[]}
       */
      specialFields: [],

      /**
       * The path to the directory that contains the SVG files which are used like an
       * icon to represent each UI type
       * @type {string}
       */
      iconDir: "templates/filters/filterIcons/",

      /**
       * A single type of custom search filter that a user can select. An option
       * represents a specific Filter model type and uses that associated Filter View.
       * @typedef {Object} UIBuilderOption
       * @property {string} label - The user-facing label to show for this option
       * @property {string} modelType - The name of the filter model type that that this
       * UI builder should create. Only one is allowed. The model must be one of the six
       * filters that are allowed in a Portal "UIFilterGroupType". See
       * {@link https://github.com/DataONEorg/collections-portals-schemas/blob/master/schemas/portals.xsd}.
       * @property {string} iconFileName - The file name, including extension, of the SVG
       * icon used to represent this option
       * @property {string} description - A very brief, user-facing description of how
       * this filter works
       * @property {string[]} filterTypes - An array of one or more filter types that are
       * allowed for this interface. If none are provided then any filter type is
       * allowed. Filter types are one of the four keys defined in
       * @property {string[]} blockedFields - An array of one or more search
       * fields for which this interface should be blocked
       * {@link QueryField#filterTypesMap}, and correspond to one of the four filter
       * types that are allowed in a Collection definition. See
       * {@link https://github.com/DataONEorg/collections-portals-schemas/blob/master/schemas/collections.xsd}.
       * This property is used to help users match custom search filter UIs to
       * appropriate query fields.
       * @property {function} modelFunction - A function that takes an optional object
       * with model properties and returns an instance of a model to use for this UI
       * builder
       * @property {function} uiFunction - A function that takes the model as an argument
       * and returns the filter UI builder view for this option
       */

      /**
       * The list of UI types that a user can select from. They will appear in the
       * carousel in the order they are listed here.
       * @type {UIBuilderOption[]}
       */
      uiBuilderOptions: [
        {
          label: "Free text",
          modelType: "Filter",
          iconFileName: "filter.svg",
          description: "Allow people to search using any text they enter",
          filterTypes: ["filter"],
          blockedFields: [],
          modelFunction: function (attrs) {
            return new Filter(attrs);
          },
          uiFunction: function (model) {
            return new FilterView({
              model: model,
              mode: "uiBuilder",
            });
          },
        },
        {
          label: "Dropdown",
          modelType: "ChoiceFilter",
          iconFileName: "choice.svg",
          description:
            "Allow people to select a search term from a list of options",
          filterTypes: ["filter"],
          blockedFields: [...MetacatUI.appModel.get("querySemanticFields")],
          modelFunction: function (attrs) {
            return new ChoiceFilter(attrs);
          },
          uiFunction: function (model) {
            return new ChoiceFilterView({
              model: model,
              mode: "uiBuilder",
            });
          },
        },
        {
          label: "Year slider",
          modelType: "DateFilter",
          iconFileName: "number.svg",
          description: "Let people search for a range of years",
          filterTypes: ["dateFilter"],
          blockedFields: [...MetacatUI.appModel.get("querySemanticFields")],
          modelFunction: function (attrs) {
            return new DateFilter(attrs);
          },
          uiFunction: function (model) {
            return new DateFilterView({
              model: model,
              mode: "uiBuilder",
            });
          },
        },
        {
          label: "Toggle",
          modelType: "ToggleFilter",
          iconFileName: "toggle.svg",
          description:
            "Let people add or remove a single, specific search term",
          filterTypes: ["filter"],
          blockedFields: [...MetacatUI.appModel.get("querySemanticFields")],
          modelFunction: function (attrs) {
            return new ToggleFilter(attrs);
          },
          uiFunction: function (model) {
            return new ToggleFilterView({
              model: model,
              mode: "uiBuilder",
            });
          },
        },
      ],

      /**
       * Executed when this view is created
       * @param {object} options - A literal object of options to pass to this view
       * @property {Filter|ChoiceFilter|DateFilter|ToggleFilter} options.model - The
       * filter model to render an editor for. It must be part of a Filters collection.
       */
      initialize: function (options) {
        try {
          // Ensure the query fields are cached for limitUITypes()
          if (typeof MetacatUI.queryFields === "undefined") {
            MetacatUI.queryFields = new QueryFields();
            MetacatUI.queryFields.fetch();
          }

          if (!options || typeof options != "object") {
            var options = {};
          }

          this.editorView = options.editorView || null;

          if (!options.isNew) {
            // If this view is an editor for an existing Filter model, check that the model
            // and the Filters collection is provided.
            if (!options.model) {
              console.log(
                "A Filter model is required to render a Filter Editor View",
              );
              return;
            }
            if (!options.model.collection) {
              console.log(
                "The Filter model for a FilterEditorView must be part of a" +
                  " Filters collection",
              );
              return;
            }
            // Set the model and collection on the view
            this.model = options.model;
            this.collection = options.model.collection;
          } else {
            // If this is an editor for a new Filter model, create a default model and
            // make sure there is a Filters collection to add it to
            if (!options.collection) {
              console.log(
                "A Filters collection is required to render a " +
                  "FilterEditorView for a new Filters model.",
              );
              return;
            }
            this.model = new Filter();
            this.collection = options.collection;
            this.isNew = true;
          }
        } catch (error) {
          console.log(
            "Error creating an FilterEditorView. Error details: " + error,
          );
        }
      },

      /**
       * Render the view
       */
      render: function () {
        try {
          // Save a reference to this view
          var view = this;

          // Create and insert an "edit" or a "add filter" button for the filter.
          var buttonText = this.text.editButton,
            buttonClasses = this.classes.editButton,
            buttonIcon = "pencil";

          // Text & styling is different for the "add a new filter" button
          if (this.isNew) {
            buttonText = this.text.addFilterButton;
            buttonIcon = "plus";
            buttonClasses = buttonClasses + " btn";
            this.$el.addClass("new");
          }
          var editButton = $(
            "<a class='" +
              buttonClasses +
              "'>" +
              "<i class='icon icon-" +
              buttonIcon +
              " icon-on-left'></i> " +
              buttonText +
              "</a>",
          );
          this.$el.prepend(editButton);

          // Render the editor modal on-the-fly to make the application load faster.
          // No need to create editing modals for filters that a user doesn't edit.
          editButton.on("click", function () {
            view.renderEditorModal.call(view);
          });

          // Save a reference to this view
          this.$el.data("view", this);
          return this;
        } catch (error) {
          console.log(
            "Error rendering an FilterEditorView. Error details: " + error,
          );
        }
      },

      /**
       * Render and show the modal window that has all the components for editing a
       * filter. This is created on-the-fly because creating these modals all at once in
       * a FilterGroupsView in edit mode takes too much time.
       */
      renderEditorModal: function () {
        try {
          // Save a reference to this view
          var view = this;

          // The list of UI Filter Editor options needs to be mutable. We will save the
          // draft filter models, and the associated editor views to this list. Rewrite
          // this.uiBuilders every time the editor modal is re-rendered.
          this.uiBuilders = [];
          this.uiBuilderOptions.forEach(function (opt) {
            this.uiBuilders.push(_.clone(opt));
          }, this);

          // Create and insert the modal window that will contain the editing interface
          var modalHTML = this.template({
            classes: view.classes,
            text: view.text,
          });
          this.modalEl = $(modalHTML);
          this.$el.append(this.modalEl);

          // Start rendering the metadata field input only after the modal is shown.
          // Otherwise this step slows the rendering down, leaves too much of a delay
          // before the modal appears.
          this.modalEl.off();
          this.modalEl.on("shown", function (event) {
            view.modalEl.off("shown");
            view.renderFieldInput();
          });
          this.modalEl.modal("show");

          // Add listeners to the modal buttons save or cancel changes
          this.activateModalButtons();

          // Create and insert the "buttons" to switch filter type, and the elements
          // that will contain the UI building interfaces for each filter type.
          this.renderUIBuilders();

          // Select and render the UI Filter Editor for the filter model set on this
          // view.
          this.switchFilterType();

          // Disable any filter types that do not match the currently selected fields
          this.handleFieldChange(_.clone(view.model.get("fields")));
        } catch (error) {
          console.log(
            "There was an error rendering the modal in a FilterEditorView" +
              " Error details: " +
              error,
          );
        }
      },

      /**
       * Hide and destroy the filter editor modal window
       */
      hideModal: function () {
        try {
          var view = this;
          view.modalEl.off("hidden");
          view.modalEl.on("hidden", function () {
            view.modalEl.off();
            view.modalEl.remove();
          });
          view.modalEl.modal("hide");
        } catch (error) {
          console.log(
            "There was an error hiding the editing modal in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Find the save and cancel buttons in the editing modal window, and add listeners
       * that close the modal and update the Filters collection on save
       */
      activateModalButtons: function () {
        try {
          var view = this;
          // The buttons at the bottom of the modal
          var saveButton = this.modalEl.find("." + this.classes.saveButton),
            cancelButton = this.modalEl.find("." + this.classes.cancelButton),
            deleteButton = this.modalEl.find("." + this.classes.deleteButton);

          // Add listeners to the modal's "delete", "save", and "cancel" buttons.

          // SAVE
          saveButton.on("click", function (event) {
            // Don't allow user to save a filter with a field that doesn't have a
            // matching UI type supported yet.
            if (view.noFilterOptions) {
              view.handleErrors();
              // Switch the message from "warning" to "error" so that it's clear this is
              // the reason the user cannot save the filter
              view.showNoFilterOptionMessage(false, "error");
              return;
            }

            var results = view.createModel();

            if (results.success === false) {
              view.handleErrors(results.errors);
              return;
            }

            saveButton.off("click");
            view.hideModal();
            // Only update the collection after the modal has closed because adding a
            // new model triggers a re-render of the FilterGroupsView which interferes
            // with removing the modal.
            var oldModel = view.model;
            // Update the filter model in the parent Filters collection
            view.model = view.collection.replaceModel(oldModel, results.model);

            if (view.editorView) {
              view.editorView.showControls();
            }
          });

          // CANCEL
          cancelButton.on("click", function (event) {
            cancelButton.off("click");
            view.currentUIBuilder = null;
            view.hideModal();
          });

          // DELETE
          deleteButton.on("click", function (event) {
            deleteButton.off("click");
            view.hideModal();
            if (!view.isNew) {
              view.collection.remove(view.model);
            }
            if (view.editorView) {
              view.editorView.showControls();
            }
          });
        } catch (error) {
          console.log(
            "There was an error activating the modal buttons in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Create and insert the "buttons" to switch filter type and the elements
       * that will contain the UI building interfaces for each filter type.
       */
      renderUIBuilders: function () {
        try {
          var view = this;

          // The container for the list of filter icons that allows users to switch
          // between filter types, plus the associated instruction paragraph
          var uiBuilderChoicesContainer = this.modalEl.find(
            "." + this.classes.uiBuilderChoicesContainer,
          );

          // The container for just the icons/buttons
          var uiBuilderChoices = $("<div></div>").addClass(
            this.classes.uiBuilderChoices,
          );
          uiBuilderChoicesContainer.append(uiBuilderChoices);

          // uiBuilderCarousel will contain all of the UIBuilder views as slides
          this.uiBuilderCarousel = this.modalEl.find(
            "." + this.classes.uiBuilderContainer,
          );

          // The bootstrap carousel plugin requires the carousel slide times to be
          // contained within an inner div with the class 'carousel-inner'
          var carouselInner = $('<div class="carousel-inner"></div>');
          this.uiBuilderCarousel.append(carouselInner);

          // Create a container and button for each uiBuilder option
          this.uiBuilders.forEach(function (uiBuilder) {
            // Create a label button that allows the user to select the given UI

            // Create the button label
            var labelEl = $("<h5>" + uiBuilder.label + "</h5>").addClass(
              view.classes.uiBuilderLabel,
            );
            // Create the button
            var button = $("<div></div>")
              .addClass(view.classes.uiBuilderChoice)
              .attr("data-filter-type", uiBuilder.modelType)
              .append(labelEl);
            // Insert the uiBuilder icon SVG into the button
            var svgPath = "text!" + this.iconDir + uiBuilder.iconFileName;
            require([svgPath], function (svgString) {
              button.append(svgString);
            });
            // Add a tooltip with description to the button
            button.tooltip({
              title: uiBuilder.description,
              delay: {
                show: 900,
                hide: 50,
              },
            });
            // Insert the button into the list of uiBuilder choices
            uiBuilderChoices.append(button);
            // Create and insert the container / carousel slide. The carousel plugin
            // requires slides to have the class 'item'. Save the container to the
            // list of uiBuilder options.
            var uiBuilderContainer = $('<div class="item"></div>');
            carouselInner.append(uiBuilderContainer);

            // Add the button and container to the list of uiBuilders to make it
            // easy to switch between filter types
            uiBuilder.container = uiBuilderContainer;
            uiBuilder.button = button;
          }, this);

          // Initialize the carousel
          this.uiBuilderCarousel.addClass("slide");
          this.uiBuilderCarousel.addClass("carousel");
          this.uiBuilderCarousel.carousel({
            interval: false,
          });
          // Need active class on at least one item for carousel to work properly
          this.uiBuilderCarousel.find(".item").first().addClass("active");
        } catch (error) {
          console.log(
            "There was an error rendering the UI filter builders in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Create and insert the component that is used to edit the fields attribute of a
       * Filter Model. Save it to the view so that the selected fields can be accessed
       * on save.
       */
      renderFieldInput: function () {
        try {
          var view = this;
          var selectedFields = _.clone(view.model.get("fields"));
          view.fieldInput = new QueryFieldSelect({
            selected: selectedFields,
            inputLabel: "Select one or more metadata fields",
            excludeFields: view.excludeFields,
            addFields: view.specialFields,
            separator: view.model.get("fieldsOperator"),
          });
          view.modalEl
            .find("." + view.classes.fieldsContainer)
            .append(view.fieldInput.el);
          view.fieldInput.render();

          this.stopListening(view.fieldInput.model, "change:selected");
          this.listenTo(
            view.fieldInput.model,
            "change:selected",
            function (_model, newSelectedFields) {
              view.handleFieldChange.call(view, newSelectedFields);
            },
          );
        } catch (error) {
          console.log(
            "There was an error rendering a fields input in a FilterEditorView" +
              " Error details: " +
              error,
          );
        }
      },

      /**
       * Run whenever the user selects or removes fields from the Query Field input.
       * This function checks which filter UIs support the type of Query Field selected,
       * and then blocks or enables the UIs in the editor. This is done to help prevent
       * users from building mis-matched search filters, e.g. "Year Slider" filters with
       * text query fields.
       * @param {string[]} selectedFields The Query Field names (i.e. Solr field names)
       * of the newly selected fields
       */
      handleFieldChange: function (selectedFields) {
        try {
          var view = this;

          // Enable all UI types if no field is selected yet
          if (
            !selectedFields ||
            !selectedFields.length ||
            selectedFields[0] === ""
          ) {
            this.uiBuilders.forEach(function (uiBuilder) {
              view.allowUI(uiBuilder);
            });
            return;
          }

          // If at least one field is selected, then limit the available UI types to
          // those that match the type of Query Field.
          var type =
            MetacatUI.queryFields.getRequiredFilterType(selectedFields);

          this.uiBuilders.forEach(function (uiBuilder) {
            if (
              uiBuilder.filterTypes.includes(type) &&
              view.isBuilderAllowedForFields(uiBuilder, selectedFields)
            ) {
              view.allowUI(uiBuilder);
            } else {
              view.blockUI(uiBuilder);
            }
          });
        } catch (error) {
          console.log(
            "There was an error handling a field change in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Marks a UI builder is blocked (so that it can't be selected) and updates the
       * tooltip with text explaining that this UI can't be used with the selected
       * fields. If the UI to block is the currently selected UI, then switches to the
       * next allowed UI. If there are no UIs that are allowed, then shows a message and
       * hides all UI builders.
       * @param {UIBuilderOption} uiBuilder - The UI builder Object to block
       */
      blockUI: function (uiBuilder) {
        try {
          var view = this;
          uiBuilder.allowed = false;
          uiBuilder.button.addClass("disabled");
          uiBuilder.button.tooltip("destroy");
          uiBuilder.button.tooltip({
            title: view.text.filterNotAllowed,
            delay: {
              show: 400,
              hide: 50,
            },
          });
          // If the current UI is a blocked one...
          if (this.currentUIBuilder === uiBuilder) {
            // ... switch to the next unblocked one.
            var allowedUIBuilder = _.findWhere(this.uiBuilders, {
              allowed: true,
            });
            if (allowedUIBuilder) {
              view.switchFilterType(allowedUIBuilder.modelType);
            } else {
              // If there is no UI available, then show a message
              this.showNoFilterOptionMessage();
            }
          }
        } catch (error) {
          console.log(
            "There was an error blocking a filter UI builder in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Marks a UI builder is allowed (so that it can be selected) and updates the
       * tooltip text with the description of this UI. If it's displayed, this function
       * hides the message that indicates that there are no allowed UIs that match the
       * selected query fields.
       * @param {UIBuilderOption} uiBuilder - The UI builder Object to block
       */
      allowUI: function (uiBuilder) {
        try {
          // If at least one UI is allowed, then make sure the "no filter message" is
          // hidden.
          if (this.noFilterOptions) {
            this.hideNoFilterOptionMessage();
          }
          uiBuilder.allowed = true;
          uiBuilder.button.removeClass("disabled");
          uiBuilder.button.tooltip("destroy");
          uiBuilder.button.tooltip({
            title: uiBuilder.description,
            delay: {
              show: 900,
              hide: 50,
            },
          });
        } catch (error) {
          console.log(
            "There was an error unblocking a filter UI builder in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Hides all filter builder UIs and displays a warning message indicating that
       * there are currently no UI options that support the selected fields.
       * @param {string} message A message to show. If not set, then the string set in
       * the view's text.noFilterOption attribute is used.
       * @param {string} [type="warning"] The type of message to display (warning,
       * error, or info)
       */
      showNoFilterOptionMessage: function (message, type = "warning") {
        try {
          this.noFilterOptions = true;
          if (!message) {
            message = this.text.noFilterOption;
          }
          if (this.noFilterOptionMessageEl) {
            this.noFilterOptionMessageEl.remove();
          }
          this.noFilterOptionMessageEl = $(
            '<div class="alert alert-' + type + '">' + message + "</div>",
          );
          this.uiBuilderCarousel.hide();
          this.uiBuilderCarousel.after(this.noFilterOptionMessageEl);
        } catch (error) {
          console.log(
            "There was an error showing a message to indicate that no filter builder " +
              "UI options are allowed in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Removes the message displayed by the
       * {@link FilterEditorView#showNoFilterOptionMessage} function and un-hides all
       * the filter builder UIs.
       */
      hideNoFilterOptionMessage: function () {
        try {
          this.noFilterOptions = false;
          if (this.noFilterOptionMessageEl) {
            this.noFilterOptionMessageEl.remove();
            console.log(this.noFilterOptionMessageEl);
          }
          if (this.uiBuilderCarousel.is(":hidden")) {
            this.uiBuilderCarousel.show();
          }
        } catch (error) {
          console.log(
            "There was an error hiding a message in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Functions to run when a user clicks the "save" button in the editing modal
       * window. Creates a new Filter model with all of the new attributes that the user
       * has selected. Checks if the model is valid. If it is, then returns the model.
       * If it is not, then returns the errors.
       * @param {Object} event The click event
       * @return {Object} Returns an object with a success property set to either true
       * (if there were no errors), or false (if there were errors). If there were
       * errors, then the object also has an errors property with the errors return from
       * the Filter validate function. If there were no errors, then the object contains
       * a model property with the new Filter to be saved to the Filters collection.
       */
      createModel: function (event) {
        try {
          var selectedUI = this.currentUIBuilder,
            newModelAttrs = selectedUI.draftModel.toJSON();

          // Set the new fields
          newModelAttrs.fields = _.clone(this.fieldInput.model.get("selected"));
          // set the new fieldsOperator
          newModelAttrs.fieldsOperator = this.fieldInput.model.get("separator");

          delete newModelAttrs.objectDOM;
          delete newModelAttrs.cid;

          // The collection's model function identifies the type of model to create
          // based on the filterType attribute. Create a model before we add it to the
          // collection, so that we can make sure it's valid first, while still allowing
          // a user to press the UNDO button and not add any changes to the Filters
          // collection.
          var newModel = this.collection.model(newModelAttrs);

          // Check if the filter is valid.
          var newModelErrors = newModel.validate();
          if (newModelErrors) {
            return {
              success: false,
              errors: newModelErrors,
            };
          } else {
            return {
              success: true,
              model: newModel,
            };
          }
        } catch (error) {
          console.log(
            "There was an error updating a Filter model in a FilterEditorView" +
              " Error details: " +
              error,
          );
        }
      },

      /**
       * Shows errors in the filter editor modal window.
       * @param {object} errors An object where keys represent the Filter model
       * attribute that has an error, and the corresponding value explains the error in
       * text.
       */
      handleErrors: function (errors) {
        try {
          var view = this;

          // Show a general error message in the modal. (Don't add it twice.)
          if (view.validationErrorEl) {
            view.validationErrorEl.remove();
          }
          view.validationErrorEl = $(
            '<p class="alert alert-error">' +
              view.text.validationError +
              "</p>",
          );
          this.$el.find(".modal-body").prepend(view.validationErrorEl);

          if (errors) {
            // Show an error for the "fields" attribute (common to all Filters)
            if (errors.fields) {
              view.fieldInput.showMessage(errors.fields, "error", true);
            }

            // Show errors for the attributes specific to each Filter type
            view.currentUIBuilder.view.showValidationErrors(errors);
          }
        } catch (error) {
          console.log(
            "There was an error  in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Function that takes the event when a user clicks on one of the filter type
       * options, gets the name of the desired filter type, and passes it to the switch
       * filter function.
       * @param {object} event The click event
       */
      handleFilterIconClick: function (event) {
        try {
          // Get the new Filter Type from the click event. The name of the new Filter
          // Type is stored as a data attribute in the clicked Filter icon.
          // var filterTypeIcon =
          var newFilterType = event.currentTarget.dataset.filterType;

          // Pass the Filter Type to the switch filter function
          this.switchFilterType(newFilterType);
        } catch (error) {
          console.log(
            "There was an error handling a click event in a FilterEditorView" +
              " Error details: " +
              error,
          );
        }
      },

      /**
       * Switches the current draft Filter model to a different Filter model type.
       * Carries over any common attributes from the previously selected filter type.
       * If no filter type is provided, defaults to type of the view's model
       * @param {string} newFilterType The name of the filter type to switch to
       */
      switchFilterType: function (newFilterType) {
        try {
          var view = this;

          // Use the filter type of the model if none is provided.
          if (!newFilterType) {
            newFilterType = this.model.type;
          }

          // Get the properties of the Filter UI Editor for the new filter type.
          var uiBuilder = _.findWhere(this.uiBuilders, {
            modelType: newFilterType,
          });

          // Don't allow user to build a mis-matched filter (e.g. text filter with date
          // field)
          if (uiBuilder.allowed === false) {
            return;
          }

          // (Index is used for the carousel)
          var index = this.uiBuilders.indexOf(uiBuilder);

          // Treat the first Filter in the list of filter UI editor options as the
          // default
          if (!uiBuilder) {
            uiBuilder = this.uiBuilders[0];
            filterType = uiBuilder.modelType;
          }

          // Create an object with the properties to pass on to the new draft model
          var newModelAttrs = {};

          // If there is a currently selected UI editor, then find the common model
          // attributes that we should pass on to the new UI editor type
          if (this.currentUIBuilder) {
            newModelAttrs = this.getCommonAttributes(
              this.currentUIBuilder.draftModel,
              newFilterType,
            );
          }
          // All search filter models are UI Filter Type
          newModelAttrs.isUIFilterType = true;

          // If a UI editor has already been created for this Filter Type, then just
          // update the pre-existing draft model. This way, if a user has already
          // selected content that is specific to a filter type (e.g. choices for a
          // choiceFilter), that content will still be there when they switch back to
          // it. Otherwise, use a clone of the model set on this view. We will update
          // the actual model in the Filters collection only when the user clicks save.
          if (!uiBuilder.draftModel) {
            if (this.model.type == newFilterType) {
              uiBuilder.draftModel = this.model.clone();
            } else {
              uiBuilder.draftModel = uiBuilder.modelFunction({
                isUIFilterType: true,
              });
            }
          }

          if (Object.keys(newModelAttrs).length) {
            uiBuilder.draftModel.set(newModelAttrs);
          }
          // Save the new selection to the view
          this.currentUIBuilder = uiBuilder;

          // Find the container for this filter type
          var uiBuilderContainer = uiBuilder.container;

          // Create or update view
          this.currentUIBuilder.view = this.currentUIBuilder.uiFunction(
            uiBuilder.draftModel,
          );
          uiBuilderContainer.html(this.currentUIBuilder.view.el);
          this.currentUIBuilder.view.render();

          // Add the selected/active class to the clicked FilterTypeIcon, remove it from
          // the other icons.
          this.uiBuilders.forEach(function (uiBuilder) {
            uiBuilder.button.removeClass(view.classes.uiBuilderChoiceActive);
          });
          this.currentUIBuilder.button.addClass(
            view.classes.uiBuilderChoiceActive,
          );

          // Have the carousel slide to the selected uiBuilder container.
          this.uiBuilderCarousel.carousel(index);
        } catch (error) {
          console.log(
            "There was an error switching filter types in a FilterEditorView." +
              " Error details: " +
              error,
          );
        }
      },

      /**
       * Checks for attribute keys that are the same between a given Filter model, and a
       * new Filter model type. Returns an object of model attributes that are relevant
       * to the new Filter model type. The values for this object will be pulled from
       * the given model. objectDOM, cid, and nodeName attributes are always excluded.
       *
       * @param {Filter} filterModel A filter model
       * @param {string} newFilterType The name of the new filter model type
       *
       * @returns {Object} returns the model attributes from the given filterModel that
       * are also relevant to the new Filter model type.
       */
      getCommonAttributes: function (filterModel, newFilterType) {
        try {
          // The filter model attributes that are common to both the current Filter Model
          // and the new Filter Type that we want to create.
          var commonAttributes = {};

          // Given the newFilterType string, get the default attribute names for a new
          // model of that type.
          var uiBuilder = _.findWhere(this.uiBuilders, {
            modelType: newFilterType,
          });
          var defaultAttrs = uiBuilder.modelFunction().defaults();
          var defaultAttrNames = Object.keys(defaultAttrs);

          // Check if any of those attribute types exist in the current filter model.
          // If they do, include them in the common attributes object.
          var currentAttrs = filterModel.toJSON();
          defaultAttrNames.forEach(function (attrName) {
            var valueInDraftModel = currentAttrs[attrName];
            if (
              valueInDraftModel ||
              (valueInDraftModel === 0) | (valueInDraftModel === false)
            ) {
              commonAttributes[attrName] = valueInDraftModel;
            }
          }, this);

          // Exclude attributes that shouldn't be passed to a new model, like the
          // objectDOM and the model ID.
          delete commonAttributes.objectDOM;
          delete commonAttributes.cid;
          delete commonAttributes.nodeName;

          // Return the common attributes
          return commonAttributes;
        } catch (error) {
          console.log(
            "There was an error getting common model attributes in a FilterEditorView" +
              ". Error details: " +
              error,
          );
        }
      },

      /**
       * Determine whether a particular UIBuilder is allowed for a set of
       * search field names. For use in handleFieldChange to enable or disable
       * certain UI builders using allowUI/blockUI when the user selects
       * different search fields.
       *
       * @param {UIBuilderOption} uiBuilder The UIBuilderOption object to
       * check
       * @param {string[]} selectedFields An array of search field names to
       * look for restrictions
       *
       * @return {boolean} Whether or not the uiBuilder is allowed for all
       * of selectedFields. Returns true only if all selectedFields are
       * allowed, not just one or more.
       */
      isBuilderAllowedForFields: function (uiBuilder, selectedFields) {
        // Return true early if this uiBuilder has no blockedFields
        if (!uiBuilder.blockedFields || uiBuilder.blockedFields.length == 0) {
          return true;
        }

        // Check each blockedField for presence in selectedFields
        var isAllowed = uiBuilder.blockedFields.map(function (blockedField) {
          return !selectedFields.includes(blockedField);
        });

        return isAllowed.every(function (e) {
          return e;
        });
      },
    },
  );
  return FilterEditorView;
});