Source: src/js/models/metadata/eml211/EMLAnnotation.js

  1. define(["jquery", "underscore", "backbone"], function ($, _, Backbone) {
  2. /**
  3. * @class EMLAnnotation
  4. * @classdesc Stores EML SemanticAnnotation elements.
  5. * @classcategory Models/Metadata/EML211
  6. * @see https://eml.ecoinformatics.org/eml-2.2.0/eml-semantics.xsd
  7. * @extends Backbone.Model
  8. */
  9. var EMLAnnotation = Backbone.Model.extend(
  10. /** @lends EMLAnnotation.prototype */ {
  11. type: "EMLAnnotation",
  12. defaults: function () {
  13. return {
  14. isNew: true,
  15. propertyLabel: null,
  16. propertyURI: null,
  17. valueLabel: null,
  18. valueURI: null,
  19. objectDOM: null,
  20. objectXML: null,
  21. };
  22. },
  23. initialize: function (attributes, opions) {
  24. this.stopListening(this, "change", this.trickleUpChange);
  25. this.listenTo(this, "change", this.trickleUpChange);
  26. },
  27. parse: function (attributes, options) {
  28. // If parsing, this is an existing annotation so it's not isNew
  29. attributes.isNew = false;
  30. var propertyURI = $(attributes.objectDOM).find("propertyuri");
  31. var valueURI = $(attributes.objectDOM).find("valueuri");
  32. if (propertyURI.length !== 1 || valueURI.length !== 1) {
  33. return;
  34. }
  35. attributes.propertyURI = $(propertyURI).text().trim();
  36. attributes.valueURI = $(valueURI).text().trim();
  37. var propertyLabel = $(propertyURI).attr("label");
  38. var valueLabel = $(valueURI).attr("label");
  39. if (!propertyLabel || !valueLabel) {
  40. return;
  41. }
  42. attributes.propertyLabel = propertyLabel.trim();
  43. attributes.valueLabel = valueLabel.trim();
  44. return attributes;
  45. },
  46. validate() {
  47. const errors = [];
  48. if (this.isEmpty()) {
  49. this.trigger("valid");
  50. return null;
  51. }
  52. const isCanonicalDataset = this.get("isCanonicalDataset");
  53. const emptyErrorMsg = (label) => `${label} must be set.`;
  54. const uriErrorMsg = (label) =>
  55. `${label} should be an HTTP(S) URI, for example: https://doi.org/xxxx.`;
  56. const isValidURI = (uri) => uri.match(/http[s]?:\/\/.+/) !== null;
  57. // Both URIs must be set and must be valid URIs
  58. const uriAttrs = [
  59. { attr: "propertyURI", label: "Property URI" },
  60. { attr: "valueURI", label: "Value URI" },
  61. ];
  62. uriAttrs.forEach(({ attr, label }) => {
  63. const uri = this.get(attr);
  64. if (!uri || uri.length <= 0) {
  65. errors.push({
  66. attr,
  67. message: emptyErrorMsg(label),
  68. isCanonicalDataset,
  69. });
  70. } else if (!isValidURI(uri)) {
  71. errors.push({
  72. attr,
  73. message: uriErrorMsg(label),
  74. isCanonicalDataset,
  75. });
  76. }
  77. });
  78. // Both labels must be set to a string
  79. const labelAttrs = [
  80. { attr: "propertyLabel", label: "Property Label" },
  81. { attr: "valueLabel", label: "Value Label" },
  82. ];
  83. labelAttrs.forEach(({ attr, label }) => {
  84. const value = this.get(attr);
  85. if (!value || value.length <= 0) {
  86. errors.push({
  87. attr,
  88. message: emptyErrorMsg(label),
  89. isCanonicalDataset,
  90. });
  91. }
  92. });
  93. if (errors.length === 0) {
  94. this.trigger("valid");
  95. return null;
  96. }
  97. return errors;
  98. },
  99. updateDOM: function (objectDOM) {
  100. objectDOM = document.createElement("annotation");
  101. if (this.get("propertyURI")) {
  102. var propertyURIEl = document.createElement("propertyuri");
  103. $(propertyURIEl).html(this.get("propertyURI"));
  104. if (this.get("propertyLabel")) {
  105. $(propertyURIEl).attr("label", this.get("propertyLabel"));
  106. }
  107. $(objectDOM).append(propertyURIEl);
  108. }
  109. if (this.get("valueURI")) {
  110. var valueURIEl = document.createElement("valueuri");
  111. $(valueURIEl).html(this.get("valueURI"));
  112. if (this.get("valueLabel")) {
  113. $(valueURIEl).attr("label", this.get("valueLabel"));
  114. }
  115. $(objectDOM).append(valueURIEl);
  116. }
  117. return objectDOM;
  118. },
  119. formatXML: function (xmlString) {
  120. return DataONEObject.prototype.formatXML.call(this, xmlString);
  121. },
  122. /**
  123. * isEmpty
  124. *
  125. * Check whether the model's properties are all empty for the purpose
  126. * of skipping the model during serialization to avoid invalid EML
  127. * documents.
  128. *
  129. * @return {boolean} - Returns true if all child elements have no
  130. * content
  131. */
  132. isEmpty: function () {
  133. return (
  134. (typeof this.get("propertyLabel") !== "string" ||
  135. this.get("propertyLabel").length <= 0) &&
  136. (typeof this.get("propertyURI") !== "string" ||
  137. this.get("propertyURI").length <= 0) &&
  138. (typeof this.get("valueLabel") !== "string" ||
  139. this.get("valueLabel").length <= 0) &&
  140. (typeof this.get("valueURI") !== "string" ||
  141. this.get("valueURI").length <= 0)
  142. );
  143. },
  144. /* Let the top level package know of attribute changes from this object */
  145. trickleUpChange: function () {
  146. MetacatUI.rootDataPackage.packageModel?.set("changed", true);
  147. },
  148. },
  149. );
  150. return EMLAnnotation;
  151. });