Source: src/js/models/CitationModel.js

"use strict";

define(["jquery", "underscore", "backbone", "collections/Citations"], (
  $,
  _,
  Backbone,
  Citations,
) => {
  /**
   * @class CitationModel
   * @classdesc A Citation Model represents a single Citation Object returned by
   * the metrics-service. A citation model can alternatively be populated with a
   * SolrResultsModel or a DataONEObjectModel, or an extension of either of
   * those models. A Citation Model can represent a citation to a local
   * MetacatUI object, or an external document or publication.
   * @classcategory Models
   * @augments Backbone.Model
   * @see https://app.swaggerhub.com/apis/nenuji/data-metrics
   */
  const Citation = Backbone.Model.extend(
    /** @lends CitationModel.prototype */ {
      /**
       * The name of this type of model
       * @type {string}
       */
      type: "CitationModel",

      /**
       * The default Citation fields
       * @name CitationModel#defaults
       * @type {object}
       * @property {string} origin - text of authors who published the source
       * dataset / document / article
       * @property {string[]} originArray - array of authors who published the
       * source dataset / document / article. Same as origin, but split on commas
       * and trimmed.
       * @property {string} title - Title of the source dataset / document /
       * article
       * @property {number} year_of_publishing - Year in which the source dataset
       * / document / article was published
       * @property {string} source_url - URL to the source dataset / document /
       * article. This is usually an external publication that cites one or more
       * DataONE datasets.
       * @property {string} source_id - Unique identifier for the source dataset /
       * document / article that cited the target dataset. This is usually an
       * external publication that cites one or more DataONE datasets.
       * @property {string} target_id - Unique identifier to the target DATAONE
       * dataset. This is the dataset that was cited by the "source" document.
       * @property {string} publisher - Publisher for the source dataset /
       * document / article
       * @property {string} journal - The journal where the the document was
       * published
       * @property {number|string} volume - The volume of the journal where the
       * document was published
       * @property {number} page - The page of the journal where the document was
       * published
       * @property {Citations} citationMetadata - When this Citation Model refers
       * to an external document, citationMetadata is a collection of DataONE
       * datasets that the external document cites. This info is retrieved by the
       * metrics service, then parsed and stored as a collection of Citation
       * Models. This attribute is used in the Portals view, for example, where we
       * display a list of external publications that cite the portal data. In
       * this case, each publication's citationMetadata is the list of local
       * MetacatUI data packages cited in the publication.
       * @property {Backbone.Model} sourceModel - The model to use to populate
       * this citation model. This can be a SolrResultsModel, a
       * DataONEObjectModel, or an extension of either of those models. Do not set
       * this attribute directly. Instead, use the setSourceModel() method.
       * @property {string} pid - The pid or unique identifier of the object being
       * cited.
       * @property {string} seriesId - The seriesId of the object being cited
       * @property {string} view_url - For citations that are in the local
       * MetacatUI repository, this is the URL to the metadata view page for the
       * object being cited.
       * @property {string} pid_url - If the pid is a DOI, then this is the URL to
       * the DOI landing page for the object being cited. This will automatically
       * be set when the pid attribute is set.
       * @property {string} seriesId_url - If the seriesId is a DOI, then this is
       * the URL to the DOI landing page for the object being cited. This will
       * automatically be set when the seriesId attribute is set.
       */
      defaults() {
        return {
          origin: null,
          originArray: [],
          title: null,
          year_of_publishing: null,
          source_url: null,
          source_id: null,
          target_id: null,
          publisher: null,
          journal: null,
          volume: null,
          page: null,
          citationMetadata: null,
          sourceModel: null,
          pid: null,
          seriesId: null,
          view_url: null,
          pid_url: null,
          seriesId_url: null,
        };
      },

      /**
       * Get the attribute getters for this model. "Attribute getters" are
       * functions that return the value of an attribute for this Citation Model
       * given a source model. The source model can be a SolrResultsModel, a
       * DataONEObjectModel, or an extension of either of those models.
       * @returns {object} - An object that maps the name of the CitationModel
       * attribute to the function that returns the value for that attribute.
       */
      attrGetters() {
        return {
          year_of_publishing: this.getYearFromSourceModel,
          title: this.getTitleFromSourceModel,
          journal: this.getJournalFromSourceModel,
          pid: this.getPidFromSourceModel,
          seriesId: this.getSeriesIdFromSourceModel,
          originArray: this.getOriginArrayFromSourceModel,
          view_url: this.getViewUrlFromSourceModel,
        };
      },

      /**
       * Override the default Backbone.Model.parse() method to convert the
       * citationMetadata object into a nested collection of CitationModels.
       * @param {object} response - The response from the metrics-service API
       * @param {object} options - Options to pass to the parse() method.
       * @returns {object} The parsed response
       */
      parse(response) {
        try {
          // strings that need formatting when coming from the metrics-service:
          const toFormat = ["journal", "page", "volume", "publisher"];
          toFormat.forEach((attr) => {
            response[attr] = this.formatMetricsServiceString(response[attr]);
          }, this);

          // Turn the author strings into CSL JSON objects
          if (response.origin) {
            const or = this.originToArray(response.origin);
          }
          let sID = response.source_id;
          if (this.isDOI(sID)) {
            if (sID.startsWith("http")) {
              sID = this.URLtoDOI(sID);
            }
            if (!sID.startsWith("doi:")) {
              sID = `doi:${sID}`;
            }
            response.source_id = sID;
          }

          // Format the citation metadata = DataONE datasets cited by this
          // citation (external document)
          const cm = response.citationMetadata;

          // We use the inline require here in addition to the define above to
          // avoid an issue caused by the circular dependency between
          // CitationModel and Citations
          const Citations = require("collections/Citations");
          if (cm) {
            if (cm && !(cm instanceof Citations)) {
              const citationMetadata = Object.entries(cm).map(([pid, data]) => {
                // Convert format from {id: {data}} to {data, id}
                const item = { ...data, pid };
                // Origin returned by metrics-service is actually an array, not a
                // string
                item.originArray = item.origin;
                delete item.origin;
                // Format the authors in the origin array
                item.originArray = item.originArray.map((author) =>
                  this.formatAuthor(author),
                );
                // Get the publish year
                const date =
                  item.datePublished || item.dateUpdated || item.dateModified;
                item.year_of_publishing = date
                  ? new Date(date).getUTCFullYear()
                  : null;
                // Because the citation metadata is always referencing an object
                // in the local MetacatUI repository, we assume that the view_url
                // exists for the given PID.
                // DOIs from the metrics service are not prefixed with "doi:"
                if (this.isDOI(pid) && !pid.startsWith("doi:")) {
                  pid = `doi:${pid}`;
                }
                item.pid = pid;
                item.view_url = `${MetacatUI.root}/view/${encodeURIComponent(pid)}`;

                return item;
              });
              response.citationMetadata = new Citations(citationMetadata);
            }
          }
          return response;
        } catch (error) {
          console.log(
            "Error parsing a CitationModel. Returning response as-is.",
            error,
          );
          return response;
        }
      },

      /**
       * Override the default Backbone.Model.set() method to format the title,
       * page, and volume attributes before setting them, and ensure that
       * attributes that are different formats of the same value are in sync,
       * including: origin and originArray; pid and pid_url; seriesId and
       * seriesId_url. This method will prevent the sourceModel attribute from
       * being set here.
       * @param {string | object} key - The attribute name to set, or an object of
       * attribute names and values to set.
       * @param {string | number | object} val - The value to set the attribute to.
       * @param {object} options - Options to pass to the set() method.
       * @see https://backbonejs.org/#Model-set
       * @since 2.23.0
       */
      set(key, val, options) {
        try {
          if (key == null) return this;

          // Handle both `"key", value` and `{key: value}` -style arguments.
          let attrs = {};
          if (typeof key === "object") {
            attrs = key;
            options = val;
          } else {
            (attrs = {})[key] = val;
          }

          // Don't allow setting the sourceModel attribute here.
          // TODO: how to handle this better?
          delete attrs.sourceModel;

          // If the title attribute is being set, then format it first
          if (Object.keys(attrs).includes("title")) {
            attrs.title = this.formatTitle(attrs.title);
          }

          // Ensure origin and originArray contain the same data, with preference
          // given to originArray. If originArray has content, then overwrite
          // origin with the a string created from originArray. If *only* origin
          // has content, then overwrite originArray with an array created from
          // origin.
          if (
            Object.keys(attrs).includes("originArray") ||
            Object.keys(attrs).includes("origin")
          ) {
            const strToArray = this.originToArray(attrs.origin);
            const arrayToStr = this.originArrayToString(attrs.originArray);
            if (arrayToStr) {
              attrs.origin = arrayToStr;
            } else {
              attrs.originArray = strToArray;
            }
          }

          // Ensure that the pid_url and seriesId_url attributes match the pid and
          // seriesId attributes being set, and vice versa. If they don't match,
          // then set them to the correct values. Prefer the content of the IDs
          // over the URLs.
          const idToUrlAttrs = [
            { id: "pid", url: "pid_url" },
            { id: "seriesId", url: "seriesId_url" },
          ];

          idToUrlAttrs.forEach(({ id, url }) => {
            if (
              Object.keys(attrs).includes(id) ||
              Object.keys(attrs).includes(url)
            ) {
              if (!!attrs[id] && !attrs[url]) {
                attrs[url] = this.DOItoURL(attrs[id]);
              } else if (!!attrs[url] && !attrs[id]) {
                attrs[id] = this.URLtoDOI(attrs[url]);
              } else if (!!attrs[id] && !!attrs[url]) {
                attrs[url] = this.DOItoURL(attrs[id]);
              }
            }
          });

          // If citationMetadata is being changed, remove old listeners and add
          // new ones
          if (Object.keys(attrs).includes("citationMetadata")) {
            if (this.citationMetadata) {
              this.stopListening(this.citationMetadata);
            }
            if (attrs.citationMetadata && attrs.citationMetadata.length) {
              this.listenTo(
                attrs.citationMetadata,
                "update",
                this.trigger.bind(this, "change"),
              );
            }
          }

          // Set modified attributes in the regular Backbone way
          Backbone.Model.prototype.set.call(this, attrs, options);
        } catch (error) {
          console.log(
            "Error in custom set() method on CitationModel. Will attempt to set" +
              " using with Backbone set(). Attributes and error stack trace:",
            { key, val, options },
            error,
          );
          Backbone.Model.prototype.set.call(this, key, val, options);
        }
      },

      /**
       * Sets the sourceModel attribute and calls the method to populate the
       * Citation Model with the sourceModel attributes. Also removes any existing
       * listeners on the previous sourceModel and readds them to the new
       * sourceModel. Use this method to set or change the sourceModel attribute.
       * @param {Backbone.Model} newSourceModel - The new sourceModel
       * @since 2.23.0
       */
      setSourceModel(newSourceModel) {
        try {
          newSourceModel =
            newSourceModel && newSourceModel.type == "Package"
              ? newSourceModel.getMetadata()
              : newSourceModel;

          // Remove any existing listeners on the previous sourceModel
          const currentSourceModel = this.get("sourceModel");
          if (currentSourceModel) {
            this.stopListening(currentSourceModel);
            const creators = currentSourceModel.get("creator") || [];
            creators.forEach((creator) => this.stopListening(creator), this);
          }

          // Add listeners to the new sourceModel
          if (newSourceModel) {
            const creatorEvents =
              "change:individualName change:organizationName change:positionName";
            const sourceModelEvents =
              "change:origin change:creator change:pubDate change:dateUploaded change:title change:seriesId change:id change:datasource";
            const creators = newSourceModel.get("creator") || [];
            this.listenTo(newSourceModel, sourceModelEvents, () => {
              this.setSourceModel(newSourceModel);
            });
            creators.forEach((creator) => {
              this.listenTo(creator, creatorEvents, () => {
                this.setSourceModel(newSourceModel);
              });
            });
          }
          Backbone.Model.prototype.set.call(
            this,
            "sourceModel",
            newSourceModel,
          );
          this.populateFromModel(newSourceModel);
        } catch (error) {
          console.log("Error in CitationModel.setSourceModel(). Error:", error);
        }
      },

      /**
       * Do not call this method directly. Instead, call setSourceModel(), which
       * will update listeners and then call this method. This method will
       * populate this citation model's attributes from another model, such as a
       * SolrResult model or a DataONEObject model. This will reset and overwrite
       * any existing attributes on this model.
       * @param {Backbone.Model} model - The model to populate from, accepts
       * SolrResult or a model that is a DataONEObject or an extended
       * DataONEObject. If no model is passed, then the model will be reset to the
       * default attributes.
       * @param newSourceModel
       * @since 2.23.0
       */
      populateFromModel(newSourceModel) {
        try {
          // Populate this model from the new sourceModel

          const newAttrs = this.defaults();

          if (!newSourceModel) {
            this.set(newAttrs);
            return;
          }

          const attrGetters = this.attrGetters();

          Object.entries(attrGetters).forEach(([attrName, getter]) => {
            const attrValue = getter.call(this, newSourceModel);
            if (attrValue) newAttrs[attrName] = attrValue;
          });

          this.set(newAttrs);
        } catch (error) {
          console.log(
            "Error populating a CitationModel from the model: ",
            newSourceModel,
            " Error: ",
            error,
          );
        }
      },

      /**
       * Get the year from the sourceModel. First look for pubDate, then
       * dateUploaded (both in SolrResult & ScienceMetadata/EML models). Lastly
       * check datePublished (found in ScienceMetadata/EML models only.)
       * @param {Backbone.Model} sourceModel - The model to get the year from
       * @returns {number} - The year
       * @since 2.23.0
       */
      getYearFromSourceModel(sourceModel) {
        try {
          const year =
            this.yearFromDate(sourceModel.get("pubDate")) ||
            this.yearFromDate(sourceModel.get("dateUploaded")) ||
            this.yearFromDate(sourceModel.get("datePublished"));
          return year;
        } catch (error) {
          console.log(
            "Error getting year from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().year_of_publishing;
        }
      },

      /**
       * Get the title from the sourceModel
       * @param {Backbone.Model} sourceModel - The model to get the title from
       * @returns {string} - The title
       * @since 2.23.0
       */
      getTitleFromSourceModel(sourceModel) {
        try {
          let title = sourceModel.get("title");
          title = Array.isArray(title) ? title[0] : title;
          // If this is a Data object, there may not be a title, so try to get the
          // title from the file name
          if (!title && sourceModel.get("fileName")) {
            let fn = sourceModel.get("fileName");
            const extRegex = /\.[^/.]+$/;
            // Save the extension
            let ext = fn ? fn.match(extRegex) : null;
            // remove the period and make it all uppercase
            ext = ext ? ext[0].replace(".", "").toUpperCase() : ext;
            // Remove the extension and replace underscores with spaces
            fn = fn.replace(extRegex, "").replace(/_+/g, " ");
            title = fn || title;
            title = title && ext ? `${title} [${ext}]` : title;
          }
          return title;
        } catch (error) {
          console.log(
            "Error getting title from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().title;
        }
      },

      /**
       * Get the journal (datasource/node) from the sourceModel. If there is a
       * datasource attribute on the sourceModel, then get the name of the member
       * node that has that datasource ID. If we can't find a member node that
       * matches the datasource, then check if the datasource is the current node.
       * If it is, then use the repository name. If there is no datasource
       * attribute, then use the current member node's name.
       * @param {Backbone.Model} sourceModel - The model to get the journal from
       * @returns {string} - The journal
       * @since 2.23.0
       */
      getJournalFromSourceModel(sourceModel) {
        try {
          let journal = null;
          const datasource = sourceModel.get("datasource");
          const mn = MetacatUI.nodeModel.getMember(datasource);
          const currentMN = MetacatUI.nodeModel.get("currentMemberNode");
          if (datasource) {
            if (mn) {
              journal = mn.name;
            } else if (datasource == MetacatUI.appModel.get("nodeId")) {
              journal = MetacatUI.appModel.get("repositoryName");
            }
          }
          if (!journal && currentMN) {
            const mnCurrent = MetacatUI.nodeModel.getMember(currentMN);
            journal = mnCurrent ? mnCurrent.name : null;
          }
          return journal;
        } catch (error) {
          console.log(
            "Error getting journal from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().journal;
        }
      },

      /**
       * Get the array of authors ("origin") from the sourceModel. First look for
       * creator (EML), then origin (science metadata & solr results), then
       * rightsHolder & submitter (base D1 object model). Convert EML parties to
       * strings & check for incorrectly escaped characters.
       * @param {Backbone.Model} sourceModel - The model to get the originArray
       * from
       * @returns {Array} - The originArray
       * @since 2.23.0
       */
      getOriginArrayFromSourceModel(sourceModel) {
        try {
          // AUTHORS
          let authors =
            // If it's an EML document, there will be a creator field
            sourceModel.get("creator") ||
            // If it's a science metadata model or solr results, use origin
            sourceModel.get("origin") ||
            "";

          // otherwise, this is probably a base D1 object model. Don't use
          // rightsHolder or submitter for now, because it might not always be the
          // author.

          // sourceModel.get("rightsHolder") ||
          // sourceModel.get("submitter");

          // Convert EML parties to strings & check for incorrectly escaped
          // characters
          if (authors) {
            authors = Array.isArray(authors) ? authors : [authors];
            authors = authors.map((author) => this.formatAuthor(author));
          }
          return authors;
        } catch (error) {
          console.log(
            "Error getting originArray from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().originArray;
        }
      },

      /**
       * Get the pid from the sourceModel. First look for id, then identifier.
       * @param {Backbone.Model} sourceModel - The model to get the pid from
       * @returns {string} - The pid
       * @since 2.23.0
       */
      getPidFromSourceModel(sourceModel) {
        try {
          const pid =
            sourceModel.get("id") || sourceModel.get("identifier") || null;
          return pid;
        } catch (error) {
          console.log(
            "Error getting the pid from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().pid;
        }
      },

      /**
       * Get the seriesId from the sourceModel. Simply looks for the seriesId
       * attribute.
       * @param {Backbone.Model} sourceModel - The model to get the seriesId from
       * @returns {string} - The seriesId
       * @since 2.23.0
       */
      getSeriesIdFromSourceModel(sourceModel) {
        try {
          const seriesId = sourceModel.get("seriesId") || null;
          return seriesId;
        } catch (error) {
          console.log(
            "Error getting the seriesId from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().seriesId;
        }
      },

      /**
       * Use the sourceModel's createViewURL() method to get the viewUrl for the
       * citation. This method is built into DataONEObject models, SolrResult
       * models, as  well as Portal models. If the sourceModel doesn't have a
       * createViewURL() method, then use the default viewUrl (null)
       * @param {Backbone.Model} sourceModel - The model to get the viewUrl from
       * @returns {string} - The viewUrl, or null if the sourceModel doesn't have
       * a createViewURL() method.
       * @since 2.23.0
       */
      getViewUrlFromSourceModel(sourceModel) {
        try {
          if (sourceModel && sourceModel.createViewURL) {
            return sourceModel.createViewURL();
          }
          return this.defaults().viewUrl;
        } catch (error) {
          console.log(
            "Error getting the viewUrl from the sourceModel. Model and error:",
            sourceModel,
            error,
          );
          return this.defaults().viewUrl;
        }
      },

      /**
       * Format an individual author for display within a citation.
       * @param {string|EMLParty} author The author to format
       * @returns {string} Returns the author as a string if it was an EMLParty
       * with any incorrectly escaped characters corrected.
       */
      formatAuthor(author) {
        try {
          // Update the origin array asynchonously if the author is an ORCID
          if (this.isOrcid(author)) this.originArrayFromOrcid(author);

          // If author is an EMLParty model, then convert it to a string with
          // given name + sur name, or organization name
          if (typeof author.toCSLJSON === "function") {
            author = author.toCSLJSON();
          } else if (typeof author === "string") {
            author = this.nameStrToCSLJSON(author);
          }

          return author;
        } catch (error) {
          console.log(
            "There was an error formatting an author, returning " +
              "the author input as is.",
            error,
          );
          return author;
        }
      },

      /**
       * Cleans up the title for display within a citation. Removes a period from
       * the end of the title if it exists and trims whitespace.This method is
       * called any time a title is set on the Citation model.
       * @param {string} title The title to format
       * @returns {string} Returns the title with a period removed from the end if
       * it exists.
       * @since 2.23.0
       */
      formatTitle(title) {
        if (!title) return "";
        return title.replace(/\.+$/, "").trim();
      },

      /**
       * Cleans up the metrics service string for display within a citation.
       * Replaces "NULL" with an empty string, removes a period from the end of
       * the string if it exists, removes curly braces, and trims whitespace.
       * @param {string} str The metrics service string to format
       * @returns {string} Returns the metrics service string with "NULL" replaced
       * with an empty string.
       * @since 2.23.0
       */
      formatMetricsServiceString(str) {
        if (!str) return "";
        // The metrics service returns "NULL" if there is no data
        str = str === "NULL" ? "" : str;
        // Replace period at the end of the string
        str = str.replace(/\.+$/, "");
        // Remove curly braces
        str = str.replace(/{|}/g, "");
        // Remove any leading or trailing whitespace
        str = str.trim();
        // Check for incorrectly escaped characters, like &
        const doc = new DOMParser().parseFromString(str, "text/html");
        str = doc.body.textContent || "";
        return str;
      },

      /**
       * Convert the author string that is returned from the metrics service into
       * CSL JSON format. Author strings that come from the metrics service take
       * many formats, which might include full given and last names, middle
       * initials, first initials, etc. Here are a few example strings: "Chelsea
       * Wegner Koch", "Lee W. Cooper", "J. Wiktor", "Sei-Ichi Saitoh", "William
       * K. W. Li", "J.R. Lovvorn". Last name prefixes like "van" or "de" are
       * stored as a "non-dropping particle". See:
       * {@link https://citeproc-js.readthedocs.io/en/latest/csl-json/markup.html#name-variables}
       * @param {str} author The author string to convert
       * @param str
       * @returns {object} Returns an object with the author's name in CSL JSON
       * format.
       * @since 2.23.0
       */
      nameStrToCSLJSON(str) {
        if (!str) return null;
        const name = {};
        str = this.formatMetricsServiceString(str);

        // If the string contains one comma, then assume it is in the format "last
        // name, first name". Move the first name to the front of the string.
        if (str.split(",").length == 2) {
          const parts = str.split(",");
          str = `${parts[1].trim()} ${parts[0].trim()}`;
        }

        const parts = str
          .trim()
          .split(/\s+|\./)
          .filter((part) => part !== "");

        if (parts.length === 1) {
          name.literal = str;
          return name;
        }

        // Assume the last word is the family name
        name.family = parts.pop();

        // Any remaining lowercase words are assumed to be non-dropping particles
        const nonDroppingParticles = parts.filter((part) =>
          part.match(/^[a-z]+$/),
        );
        if (nonDroppingParticles.length > 0) {
          name["non-dropping-particle"] = nonDroppingParticles.join(" ");
        }

        // Any remaining words are assumed to be given names
        const givenNames = parts.filter((part) => !part.match(/^[a-z]+$/));
        if (givenNames.length > 0) {
          name.given = givenNames.join(" ");
        }

        return name;
      },

      /**
       * Given a date, extract the year as a number.
       * @param {Date | string | number} date The date to extract the year from
       * @returns {number} Returns the year as a number, or null if the date is
       * invalid.
       * @since 2.23.0
       */
      yearFromDate(date) {
        try {
          if (!date) return null;
          // If Date is already a year (Number object with 4 digits), return it
          if (Number.isInteger(date) && date.toString().length == 4) {
            return date;
          }
          // If it is a string with 4 digits, return it as an integer. Use regex.
          if (typeof date === "string" && /^\d{4}$/.test(date)) {
            return parseInt(date);
          }
          // Check if the date is a Date object
          if (!(date instanceof Date)) {
            date = new Date(date);
          }
          const yr = date.getUTCFullYear();
          return yr == "NaN" ? null : yr;
        } catch (error) {
          console.log(
            "There was an error getting the year from the date, returning null.",
            error,
          );
          return null;
        }
      },

      /**
       * Check if a string is a valid ORCID.
       * @param {string} orcid The ORCID to check
       * @returns {boolean} Returns true if the ORCID is valid, false otherwise
       * @since 2.23.0
       */
      isOrcid(orcid) {
        try {
          if (!orcid) return false;
          const regex = new RegExp(
            "^https?:\\/\\/orcid.org\\/(\\d{4}-){3}(\\d{3}[0-9X])$",
          );
          return regex.test(orcid);
        } catch {
          return false;
        }
      },

      /**
       * Use the App Lookup model's get Accounts method to get the name of the
       * author from their ORCID, then asynchronously set the originArray to
       * contain that name.
       * @param {string} orcid The ORCID to get the name for
       * @since 2.23.0
       */
      originArrayFromOrcid(orcid) {
        try {
          const request = { term: orcid };
          const model = this;

          const callback = function (response) {
            let name = null;
            if (response) {
              if (Array.isArray(response)) {
                const { label } = response[0];
                if (label) {
                  // Name is the format "Min Liew
                  // (http://orcid.org/0000-0002-5156-4610)" We want to return
                  // "Min Liew". It will always be two spaces and a "("
                  name = label.split("  (")[0];
                }
              }
            }
            if (name) {
              console.log("Setting originArray to ", [name]);
              model.set("originArray", [name]);
            }
          };
          MetacatUI.appLookupModel.getAccountsAutocomplete(request, callback);
        } catch (error) {
          console.log(
            "There was an error getting the name from the orcid.",
            error,
          );
        }
      },

      /**
       * Checks if the citation is for a DataONE object from a specific node (e.g.
       * PANGAEA)
       * @param {string} node - The node id to check, e.g. "urn:node:PANGAEA"
       * @returns {boolean} - True if the citation is for a DataONE object from
       * the given node
       * @since 2.23.0
       */
      isFromNode(node) {
        try {
          const sourceModel = this.get("sourceModel");
          return (
            sourceModel &&
            sourceModel.get &&
            sourceModel.get("datasource") &&
            sourceModel.get("datasource") === node
          );
        } catch (error) {
          console.log(
            `There was an error checking if the citation is from node ${node}.` +
              `Returning false.`,
            error,
          );
          return false;
        }
      },

      /**
       * Convert the comma-separated origin string to an array of authors.
       * @param {string} origin - The origin string to convert to an array. If a
       * falsy value is passed in, then the default originArray attribute of the
       * model is returned.
       * @returns {Array} - An array of authors
       * @since 2.23.0
       */
      originToArray(origin) {
        try {
          if (!origin) {
            return this.defaults().originArray;
          }
          const originArray = origin ? origin.split(", ") : [];
          return originArray.map((author) => this.formatAuthor(author));
        } catch (error) {
          console.log(
            "There was an error converting the origin string to an array.",
            error,
          );
          return this.defaults().originArray;
        }
      },

      /**
       * Convert the origin array to a string.
       * @param originArray
       * @returns {string} - The origin string. If a falsy value is passed in,
       * then the default origin attribute of the model is returned.
       * @since 2.23.0
       */
      originArrayToString(originArray) {
        try {
          if (!originArray || !originArray.length) {
            return this.defaults().origin;
          }
          // Each author in the array is a CSL JSON object. Map it to a string
          // with the author's name in the format First Particle Last or Literal.
          // Separate each author name with a comma and a space.
          const origin = originArray
            .map((a) => {
              if (!a) return null;
              const ndp = a["non-dropping-particle"];
              const name =
                (a.given ? `${a.given} ` : "") +
                (ndp ? `${ndp} ` : "") +
                (a.family ? a.family : "");
              return (name || a.literal || "").trim();
            })
            .filter((a) => a)
            .join(", ");
          return origin;
        } catch (error) {
          console.log(
            "There was an error converting the origin array to a string.",
            error,
          );
          return this.defaults().origin;
        }
      },

      /**
       * Returns true if the citation is for a DataONE object that has been
       * archived and archived content is not available in the search index.
       * @returns {boolean} - True if the citation has no content because it is
       * archived and archived content is not indexed.
       * @see AppModel#archivedContentIsIndexed
       * @since 2.23.0
       */
      isArchivedAndNotIndexed() {
        return this.isArchived() && !this.archivedContentIsIndexed();
      },

      /**
       * Checks if the object being cited is archived, according to the `archived`
       * attribute of the source model.
       * @returns {boolean} - True if the source model has an `archived` attribute
       * that is true.
       * @since 2.23.0
       */
      isArchived() {
        return (
          this.sourceModel &&
          this.sourceModel.get &&
          this.sourceModel.get("archived")
        );
      },

      /**
       * Checks if archived content is available in the search index.
       * @see AppModel#archivedContentIsIndexed
       * @returns {boolean} - True if archived content is available in the search
       * index.
       * @since 2.23.0
       */
      archivedContentIsIndexed() {
        return MetacatUI.appModel.get("archivedContentIsIndexed");
      },

      /**
       * Remove all DOI prefixes from a DOI string, including https, http, doi.org,
       * dx.doi.org, and doi:.
       * @param {string} str - The DOI string to remove prefixes from.
       * @returns {string} - The DOI string without any prefixes.
       * @since 2.23.0
       */
      removeAllDOIPrefixes(str) {
        if (!str) return "";
        // Remove https and http prefixes
        str = str.replace(/^(https?:\/\/)?/, "");
        // Remove domain prefixes, like doi.org and dx.doi.org
        str = str.replace(/^(doi\.org\/|dx\.doi\.org\/)/, "");
        // Remove doi: prefix
        str = str.replace(/^doi:/, "");
        return str;
      },

      /**
       * Check if a string is a valid DOI.
       * @param {string} doi - The string to check.
       * @param str
       * @returns {boolean} - True if the string is a valid DOI, false otherwise.
       * @since 2.23.0
       */
      isDOI(str) {
        return MetacatUI.appModel.isDOI(str);
      },

      /**
       * Get the URL for the online location of the object being cited when it has
       * a DOI. If the DOI is not passed to the function, or if the string is not
       * a DOI, then an empty string is returned.
       * @param {string} [str] - The DOI string with or without the "doi:" prefix.
       * It may already be a URL, or it may be a DOI string. It also handles the
       * case where the DOI is already a URL.
       * @returns {string} - The DOI URL
       * @since 2.23.0
       */
      DOItoURL(str) {
        return MetacatUI.appModel.DOItoURL(str);
      },

      /**
       * Get the DOI from a DOI URL. The URL can be http or https, can include the
       * "doi:" prefix or not, and can use "dx.doi.org" or "doi.org" as the
       * domain. If a string is not passed to the function, or if the string is
       * not for a DOI URL, then an empty string is returned.
       * @param {string} url - The DOI URL
       * @returns {string} - The DOI string, including the "doi:" prefix
       * @since 2.23.0
       */
      URLtoDOI(url) {
        return MetacatUI.appModel.URLtoDOI(url);
      },

      /**
       * Checks if the citation has a DOI in the seriesId or pid attributes.
       * @returns {string} - The DOI of the seriesID, if it is a DOI, or the DOI
       * of the pid, if it is a DOI. Otherwise, returns null.
       * @since 2.23.0
       */
      findDOI() {
        try {
          if (!this.sourceModel || !this.sourceModel.isDOI) return null;
          const seriesID = this.get("seriesId");
          const pid = this.get("pid");
          if (this.sourceModel.isDOI(seriesID)) return seriesID;
          if (this.sourceModel.isDOI(pid)) return pid;
          return null;
        } catch (error) {
          console.log(
            "There was an error finding the DOI for the citation. Returning null",
            error,
          );
          return null;
        }
      },

      /**
       * Checks if the citation has a DOI in the seriesId or pid attributes.
       * @returns {boolean} - True if the citation has a DOI
       * @since 2.23.0
       */
      hasDOI() {
        return !!this.findDOI();
      },

      /**
       * If this citation has a source model, and if that source model is a
       * DataONEObject, then return the results of the DataONEObject's
       * `getUploadStatus` function
       * @returns {string} - The upload status of the source model, if it is a
       * DataONEObject, or null if it is not.
       * @since 2.23.0
       * @see DataONEObject#getUploadStatus
       */
      getUploadStatus() {
        return this.sourceModel ? this.sourceModel.get("uploadStatus") : null;
      },

      /**
       * Get the URL for the citation. This will check the model for the following
       * attributes and return the first that is not empty: view_url, source_url,
       * sid_url, pid_url.
       * @returns {string} Returns the URL for the citation or an empty string.
       * @since 2.23.0
       */
      getURL() {
        const urlSources = ["view_url", "source_url", "sid_url", "pid_url"];
        for (let i = 0; i < urlSources.length; i++) {
          const url = this.get(urlSources[i]);
          if (url) return url;
        }
        return "";
      },

      /**
       * Get the main identifier for the citation. This will check the model for
       * the following attributes and return the first that is not empty: pid,
       * seriesId, source_url.
       * @returns {string} Returns the main identifier for the citation or an
       * empty string.
       * @since 2.23.0
       */
      getID() {
        const idSources = ["pid", "seriesId", "source_url"];
        for (let i = 0; i < idSources.length; i++) {
          const id = this.get(idSources[i]);
          if (id) return id;
        }
        return "";
      },
    },
  );

  return Citation;
});