Source: src/js/views/metadata/EMLEntityView.js

  1. define([
  2. "underscore",
  3. "jquery",
  4. "backbone",
  5. "localforage",
  6. "models/DataONEObject",
  7. "models/metadata/eml211/EMLAttribute",
  8. "models/metadata/eml211/EMLEntity",
  9. "views/DataPreviewView",
  10. "views/metadata/EMLAttributeView",
  11. "text!templates/metadata/eml-entity.html",
  12. "text!templates/metadata/eml-attribute-menu-item.html",
  13. "common/Utilities",
  14. ], function (
  15. _,
  16. $,
  17. Backbone,
  18. LocalForage,
  19. DataONEObject,
  20. EMLAttribute,
  21. EMLEntity,
  22. DataPreviewView,
  23. EMLAttributeView,
  24. EMLEntityTemplate,
  25. EMLAttributeMenuItemTemplate,
  26. Utilities,
  27. ) {
  28. /**
  29. * @class EMLEntityView
  30. * @classdesc An EMLEntityView shows the basic attributes of a DataONEObject, as described by EML
  31. * @classcategory Views/Metadata
  32. * @screenshot views/metadata/EMLEntityView.png
  33. * @extends Backbone.View
  34. */
  35. var EMLEntityView = Backbone.View.extend(
  36. /** @lends EMLEntityView.prototype */ {
  37. tagName: "div",
  38. className: "eml-entity modal hide fade",
  39. id: null,
  40. /* The HTML template for an entity */
  41. template: _.template(EMLEntityTemplate),
  42. attributeMenuItemTemplate: _.template(EMLAttributeMenuItemTemplate),
  43. fillButtonTemplateString:
  44. '<button class="btn btn-primary fill-button"><i class="icon-magic"></i> Fill from file</button>',
  45. /**
  46. * A list of file formats that can be auto-filled with attribute information
  47. * @type {string[]}
  48. * @since 2.15.0
  49. */
  50. fillableFormats: ["text/csv"],
  51. /* Events this view listens to */
  52. events: {
  53. change: "saveDraft",
  54. "change input": "updateModel",
  55. "change textarea": "updateModel",
  56. "click .entity-container > .nav-tabs a": "showTab",
  57. "click .attribute-menu-item": "showAttribute",
  58. "mouseover .attribute-menu-item .remove": "previewAttrRemove",
  59. "mouseout .attribute-menu-item .remove": "previewAttrRemove",
  60. "click .attribute-menu-item .remove": "removeAttribute",
  61. "click .fill-button": "handleFill",
  62. },
  63. initialize: function (options) {
  64. if (!options) var options = {};
  65. this.model = options.model || new EMLEntity();
  66. this.DataONEObject = options.DataONEObject;
  67. },
  68. render: function () {
  69. this.renderEntityTemplate();
  70. this.renderPreview();
  71. this.renderAttributes();
  72. this.renderFillButton();
  73. this.listenTo(this.model, "invalid", this.showValidation);
  74. this.listenTo(this.model, "valid", this.showValidation);
  75. },
  76. renderEntityTemplate: function () {
  77. var modelAttr = this.model.toJSON();
  78. if (!modelAttr.entityName) modelAttr.title = "this data";
  79. else modelAttr.title = modelAttr.entityName;
  80. modelAttr.uniqueId = this.model.cid;
  81. this.$el.html(this.template(modelAttr));
  82. //Initialize the modal window
  83. this.$el.modal();
  84. //Set the menu height
  85. var view = this;
  86. this.$el.on("shown", function () {
  87. view.adjustHeight();
  88. view.setMenuWidth();
  89. window.addEventListener("resize", function (event) {
  90. view.adjustHeight();
  91. view.setMenuWidth();
  92. });
  93. });
  94. this.$el.on("hidden", function () {
  95. view.showValidation();
  96. });
  97. },
  98. renderPreview: function () {
  99. //Get the DataONEObject model
  100. if (this.DataONEObject) {
  101. var dataPreview = new DataPreviewView({
  102. model: this.DataONEObject,
  103. });
  104. dataPreview.render();
  105. this.$(".preview-container").html(dataPreview.el);
  106. if (dataPreview.$el.children().length) {
  107. this.$(".description").css("width", "calc(100% - 310px)");
  108. } else dataPreview.$el.remove();
  109. }
  110. },
  111. renderAttributes: function () {
  112. //Render the attributes
  113. var attributes = this.model.get("attributeList"),
  114. attributeListEl = this.$(".attribute-list"),
  115. attributeMenuEl = this.$(".attribute-menu");
  116. _.each(
  117. attributes,
  118. function (attr) {
  119. //Create an EMLAttributeView
  120. var view = new EMLAttributeView({
  121. model: attr,
  122. });
  123. //Create a link in the attribute menu
  124. var menuItem = $(
  125. this.attributeMenuItemTemplate({
  126. attrId: attr.cid,
  127. attributeName: attr.get("attributeName"),
  128. classes: "",
  129. }),
  130. ).data({
  131. model: attr,
  132. attributeView: view,
  133. });
  134. attributeMenuEl.append(menuItem);
  135. menuItem.find(".tooltip-this").tooltip();
  136. this.listenTo(attr, "change:attributeName", function (attr) {
  137. menuItem.find(".name").text(attr.get("attributeName"));
  138. });
  139. view.render();
  140. attributeListEl.append(view.el);
  141. view.$el.hide();
  142. this.listenTo(attr, "change", this.addAttribute);
  143. this.listenTo(attr, "invalid", this.showAttributeValidation);
  144. this.listenTo(attr, "valid", this.hideAttributeValidation);
  145. },
  146. this,
  147. );
  148. //Add a new blank attribute view at the end
  149. this.addNewAttribute();
  150. //If there are no attributes in this EML model yet,
  151. //then make sure we show a new add attribute button when the user starts typing
  152. if (attributes.length == 0) {
  153. var onlyAttrView = this.$(".attribute-menu-item")
  154. .first()
  155. .data("attributeView"),
  156. view = this,
  157. keyUpCallback = function () {
  158. //This attribute is no longer new
  159. view.$(".attribute-menu-item.new").first().removeClass("new");
  160. view
  161. .$(".attribute-list .eml-attribute.new")
  162. .first()
  163. .removeClass("new");
  164. //Add a new attribute link and view
  165. view.addNewAttribute();
  166. //Don't listen to keyup anymore
  167. onlyAttrView.$el.off("keyup", keyUpCallback);
  168. };
  169. onlyAttrView.$el.on("keyup", keyUpCallback);
  170. }
  171. //Activate the first navigation item
  172. var firstAttr = this.$(".side-nav-item").first();
  173. firstAttr.addClass("active");
  174. //Show the first attribute view
  175. firstAttr.data("attributeView").$el.show();
  176. firstAttr.data("attributeView").postRender();
  177. },
  178. renderFillButton: function () {
  179. var formatGuess = this.model.get("dataONEObject")
  180. ? this.model.get("dataONEObject").get("formatId")
  181. : this.model.get("entityType");
  182. if (!_.contains(this.fillableFormats, formatGuess)) {
  183. return;
  184. }
  185. var target = this.$(".fill-button-container");
  186. if (!target.length === 1) {
  187. return;
  188. }
  189. var btn = $(this.fillButtonTemplateString);
  190. $(target).html(btn);
  191. },
  192. updateModel: function (e) {
  193. var changedAttr = $(e.target).attr("data-category");
  194. if (!changedAttr) return;
  195. var emlModel = this.model.getParentEML(),
  196. newValue = emlModel
  197. ? emlModel.cleanXMLText($(e.target).val())
  198. : $(e.target).val();
  199. this.model.set(changedAttr, newValue);
  200. this.model.trickleUpChange();
  201. },
  202. addNewAttribute: function () {
  203. //Check if there is already a new attribute view
  204. if (this.$(".attribute-list .eml-attribute.new").length) {
  205. return;
  206. }
  207. var newAttrModel = new EMLAttribute({
  208. parentModel: this.model,
  209. xmlID: DataONEObject.generateId(),
  210. }),
  211. newAttrView = new EMLAttributeView({
  212. isNew: true,
  213. model: newAttrModel,
  214. });
  215. newAttrView.render();
  216. this.$(".attribute-list").append(newAttrView.el);
  217. newAttrView.$el.hide();
  218. //Change the last menu item if it still says "Add attribute"
  219. if (this.$(".attribute-menu-item").length == 1) {
  220. var firstAttrMenuItem = this.$(".attribute-menu-item").first();
  221. if (firstAttrMenuItem.find(".name").text() == "Add attribute") {
  222. firstAttrMenuItem.find(".name").text("New attribute");
  223. firstAttrMenuItem.find(".add").hide();
  224. }
  225. }
  226. //Create the new menu item
  227. var menuItem = $(
  228. this.attributeMenuItemTemplate({
  229. attrId: newAttrModel.cid,
  230. attributeName: "Add attribute",
  231. classes: "new",
  232. }),
  233. ).data({
  234. model: newAttrModel,
  235. attributeView: newAttrView,
  236. });
  237. menuItem.find(".add").show();
  238. this.$(".attribute-menu").append(menuItem);
  239. menuItem.find(".tooltip-this").tooltip();
  240. //When the attribute name is changed, update the navigation
  241. this.listenTo(newAttrModel, "change:attributeName", function (attr) {
  242. menuItem.find(".name").text(attr.get("attributeName"));
  243. menuItem.find(".add").hide();
  244. });
  245. this.listenTo(newAttrModel, "change", this.addAttribute);
  246. this.listenTo(newAttrModel, "invalid", this.showAttributeValidation);
  247. this.listenTo(newAttrModel, "valid", this.hideAttributeValidation);
  248. },
  249. addAttribute: function (emlAttribute) {
  250. //Add the attribute to the attribute list in the EMLEntity model
  251. if (!_.contains(this.model.get("attributeList"), emlAttribute))
  252. this.model.addAttribute(emlAttribute);
  253. },
  254. removeAttribute: function (e) {
  255. var removeBtn = $(e.target);
  256. var menuItem = removeBtn.parents(".attribute-menu-item"),
  257. attrModel = menuItem.data("model");
  258. if (attrModel) {
  259. //Remove the attribute from the model
  260. this.model.removeAttribute(attrModel);
  261. //If this menu item is active, then make the next attribute active instead
  262. if (menuItem.is(".active")) {
  263. var nextMenuItem = menuItem.next();
  264. if (!nextMenuItem.length || nextMenuItem.is(".new")) {
  265. nextMenuItem = menuItem.prev();
  266. }
  267. if (nextMenuItem.length) {
  268. nextMenuItem.addClass("active");
  269. this.showAttribute(nextMenuItem.data("model"));
  270. }
  271. }
  272. //Remove the elements for this attribute from the page
  273. menuItem.remove();
  274. this.$(
  275. ".eml-attribute[data-attribute-id='" + attrModel.cid + "']",
  276. ).remove();
  277. $(".tooltip").remove();
  278. this.model.trickleUpChange();
  279. }
  280. },
  281. adjustHeight: function (e) {
  282. var contentAreaHeight =
  283. this.$(".modal-body").height() -
  284. this.$(".entity-container .nav-tabs").height();
  285. this.$(".attribute-menu, .attribute-list").css(
  286. "height",
  287. contentAreaHeight + "px",
  288. );
  289. },
  290. setMenuWidth: function () {
  291. this.$(".entity-container .nav").width(this.$el.width());
  292. },
  293. /**
  294. * Shows the attribute in the attribute editor
  295. * @param {Event} e - JS event or attribute model
  296. */
  297. showAttribute: function (e) {
  298. if (e.target) {
  299. var clickedEl = $(e.target),
  300. menuItem =
  301. clickedEl.is(".attribute-menu-item") ||
  302. clickedEl.parents(".attribute-menu-item");
  303. if (clickedEl.is(".remove")) return;
  304. } else {
  305. var menuItem = this.$(
  306. ".attribute-menu-item[data-attribute-id='" + e.cid + "']",
  307. );
  308. }
  309. if (!menuItem) return;
  310. //Validate the previously edited attribute
  311. //Get the current active attribute
  312. var activeAttrTab = this.$(".attribute-menu-item.active");
  313. //If there is a currently-active attribute tab,
  314. if (activeAttrTab.length) {
  315. //Get the attribute list from this view's model
  316. var emlAttributes = this.model.get("attributeList");
  317. //If there is an EMLAttribute list,
  318. if (emlAttributes && emlAttributes.length) {
  319. //Get the active EMLAttribute
  320. var activeEMLAttribute = _.findWhere(emlAttributes, {
  321. cid: activeAttrTab.attr("data-attribute-id"),
  322. });
  323. //If there is an active EMLAttribute model, validate it
  324. if (activeEMLAttribute) {
  325. activeEMLAttribute.isValid();
  326. }
  327. }
  328. }
  329. //If the user clicked on the add attribute link
  330. if (
  331. menuItem.is(".new") &&
  332. this.$(".new.attribute-menu-item").length < 2
  333. ) {
  334. //Change the attribute menu item
  335. menuItem.removeClass("new").find(".name").text("New attribute");
  336. this.$(".eml-attribute.new").removeClass("new");
  337. menuItem.find(".add").hide();
  338. //Add a new attribute view and menu item
  339. this.addNewAttribute();
  340. //Scroll the attribute menu to the bottom so that the "Add New" button is always visible
  341. var attrMenuHeight =
  342. this.$(".attribute-menu").scrollTop() +
  343. this.$(".attribute-menu").height();
  344. this.$(".attribute-menu").scrollTop(attrMenuHeight);
  345. }
  346. //Get the attribute view
  347. var attrView = menuItem.data("attributeView");
  348. //Change the active attribute in the menu
  349. this.$(".attribute-menu-item.active").removeClass("active");
  350. menuItem.addClass("active");
  351. //Hide the old attribute view
  352. this.$(".eml-attribute").hide();
  353. //Show the new attribute view
  354. attrView.$el.show();
  355. //Scroll to the top of the attribute view
  356. this.$(".attribute-list").scrollTop(0);
  357. attrView.postRender();
  358. },
  359. /**
  360. * Show the attribute validation errors in the attribute navigation menu
  361. * @param {EMLAttribute} attr
  362. */
  363. showAttributeValidation: function (attr) {
  364. var attrLink = this.$(
  365. ".attribute-menu-item[data-attribute-id='" + attr.cid + "']",
  366. ).find("a");
  367. //If the validation is already displayed, then exit
  368. if (attrLink.is(".error")) return;
  369. var errorIcon = $(document.createElement("i")).addClass(
  370. "icon icon-exclamation-sign error icon-on-left",
  371. );
  372. attrLink.addClass("error").prepend(errorIcon);
  373. },
  374. /**
  375. * Hide the attribute validation errors from the attribute navigation menu
  376. */
  377. hideAttributeValidation: function (attr) {
  378. this.$(".attribute-menu-item[data-attribute-id='" + attr.cid + "']")
  379. .find("a")
  380. .removeClass("error")
  381. .find(".icon.error")
  382. .remove();
  383. },
  384. /**
  385. * Show the user what will be removed when this remove button is clicked
  386. */
  387. previewAttrRemove: function (e) {
  388. var removeBtn = $(e.target);
  389. removeBtn.parents(".attribute-menu-item").toggleClass("remove-preview");
  390. },
  391. /**
  392. *
  393. * Will display validation styling and messaging. Should be called after
  394. * this view's model has been validated and there are error messages to display
  395. */
  396. showValidation: function () {
  397. //Reset the error messages and styling
  398. //Only change elements inside the overview-container which contains only the
  399. // EMLEntity metadata. The Attributes will be changed by the EMLAttributeView.
  400. this.$(".overview-container .notification").text("");
  401. this.$(
  402. ".overview-tab .icon.error, .attributes-tab .icon.error",
  403. ).remove();
  404. this.$(
  405. ".overview-container, .overview-tab a, .attributes-tab a, .overview-container .error",
  406. ).removeClass("error");
  407. var overviewTabErrorIcon = false,
  408. attributeTabErrorIcon = false;
  409. _.each(
  410. this.model.validationError,
  411. function (errorMsg, category) {
  412. if (category == "attributeList") {
  413. //Create an error icon for the Attributes tab
  414. if (!attributeTabErrorIcon) {
  415. var errorIcon = $(document.createElement("i"))
  416. .addClass("icon icon-on-left icon-exclamation-sign error")
  417. .attr("title", "There is missing information in this tab");
  418. //Add the icon to the Overview tab
  419. this.$(".attributes-tab a")
  420. .prepend(errorIcon)
  421. .addClass("error");
  422. }
  423. return;
  424. }
  425. //Get all the elements for this category and add the error class
  426. this.$(
  427. ".overview-container [data-category='" + category + "']",
  428. ).addClass("error");
  429. //Get the notification element for this category and add the error message
  430. this.$(
  431. ".overview-container .notification[data-category='" +
  432. category +
  433. "']",
  434. ).text(errorMsg);
  435. //Create an error icon for the Overview tab
  436. if (!overviewTabErrorIcon) {
  437. var errorIcon = $(document.createElement("i"))
  438. .addClass("icon icon-on-left icon-exclamation-sign error")
  439. .attr("title", "There is missing information in this tab");
  440. //Add the icon to the Overview tab
  441. this.$(".overview-tab a").prepend(errorIcon).addClass("error");
  442. overviewTabErrorIcon = true;
  443. }
  444. },
  445. this,
  446. );
  447. },
  448. /**
  449. * Show the entity overview or attributes tab
  450. * depending on the click target
  451. * @param {Event} e
  452. */
  453. showTab: function (e) {
  454. e.preventDefault();
  455. //Get the clicked link
  456. var link = $(e.target);
  457. //Remove the active class from all links and add it to the new active link
  458. this.$(".entity-container > .nav-tabs li").removeClass("active");
  459. link.parent("li").addClass("active");
  460. //Hide all the panes and show the correct one
  461. this.$(".entity-container > .tab-content > .tab-pane").hide();
  462. this.$(link.attr("href")).show();
  463. },
  464. /**
  465. * Show the entity in a modal dialog
  466. */
  467. show: function () {
  468. this.$el.modal("show");
  469. },
  470. /**
  471. * Hide the entity modal dialog
  472. */
  473. hide: function () {
  474. this.$el.modal("hide");
  475. },
  476. /**
  477. * Save a draft of the parent EML model
  478. */
  479. saveDraft: function () {
  480. var view = this;
  481. try {
  482. var model = this.model.getParentEML();
  483. var draftModel = model.clone();
  484. var title = model.get("title") || "No title";
  485. LocalForage.setItem(model.get("id"), {
  486. id: model.get("id"),
  487. datetime: new Date().toISOString(),
  488. title: Array.isArray(title) ? title[0] : title,
  489. draft: draftModel.serialize(),
  490. }).then(function () {
  491. view.clearOldDrafts();
  492. });
  493. } catch (ex) {
  494. console.log("Error saving draft:", ex);
  495. }
  496. },
  497. /**
  498. * Clear older drafts by iterating over the sorted list of drafts
  499. * stored by LocalForage and removing any beyond a hardcoded limit.
  500. */
  501. clearOldDrafts: function () {
  502. var drafts = [];
  503. try {
  504. LocalForage.iterate(function (value, key, iterationNumber) {
  505. // Extract each draft
  506. drafts.push({
  507. key: key,
  508. value: value,
  509. });
  510. })
  511. .then(function () {
  512. // Sort by datetime
  513. drafts = _.sortBy(drafts, function (draft) {
  514. return draft.value.datetime.toString();
  515. }).reverse();
  516. })
  517. .then(function () {
  518. _.each(drafts, function (draft, i) {
  519. var age = new Date() - new Date(draft.value.datetime);
  520. var isOld = age / 2678400000 > 1; // ~31days
  521. // Delete this draft is not in the most recent 100 or
  522. // if older than 31 days
  523. var shouldDelete = i > 100 || isOld;
  524. if (!shouldDelete) {
  525. return;
  526. }
  527. LocalForage.removeItem(draft.key).then(function () {
  528. // Item should be removed
  529. });
  530. });
  531. });
  532. } catch (ex) {
  533. console.log("Failed to clear old drafts: ", ex);
  534. }
  535. },
  536. /**
  537. * Handle the click event on the fill button
  538. *
  539. * @param {Event} e - The click event
  540. * @since 2.15.0
  541. */
  542. handleFill: function (e) {
  543. var d1Object = this.model.get("dataONEObject");
  544. if (!d1Object) {
  545. return;
  546. }
  547. var file = d1Object.get("uploadFile");
  548. try {
  549. if (!file) {
  550. this.handleFillViaFetch();
  551. } else {
  552. this.handleFillViaFile(file);
  553. }
  554. } catch (error) {
  555. console.log("Error while attempting to fill", error);
  556. view.updateFillButton(
  557. '<i class="icon-warning-sign"></i> Couldn\'t fill',
  558. );
  559. }
  560. },
  561. /**
  562. * Handle the fill event using a File object
  563. *
  564. * @param {File} file - A File object to fill from
  565. * @since 2.15.0
  566. */
  567. handleFillViaFile: function (file) {
  568. var view = this;
  569. Utilities.readSlice(file, this, function (event) {
  570. if (event.target.readyState !== FileReader.DONE) {
  571. return;
  572. }
  573. view.tryParseAndFillAttributeNames.bind(view)(event.target.result);
  574. });
  575. },
  576. /**
  577. * Handle the fill event by fetching the object
  578. * @since 2.15.0
  579. */
  580. handleFillViaFetch: function () {
  581. var view = this;
  582. var requestSettings = {
  583. url:
  584. MetacatUI.appModel.get("objectServiceUrl") +
  585. encodeURIComponent(this.model.get("dataONEObject").get("id")),
  586. method: "get",
  587. success: view.tryParseAndFillAttributeNames.bind(this),
  588. error: function (error) {
  589. view.updateFillButton(
  590. '<i class="icon-warning-sign"></i> Couldn\'t fill',
  591. );
  592. console.error(
  593. "Error fetching DataObject to parse out headers",
  594. error,
  595. );
  596. },
  597. };
  598. this.updateFillButton('<i class="icon-time"></i> Please wait...', true);
  599. this.disableFillButton();
  600. requestSettings = _.extend(
  601. requestSettings,
  602. MetacatUI.appUserModel.createAjaxSettings(),
  603. );
  604. $.ajax(requestSettings);
  605. },
  606. /**
  607. * Attempt to parse header and fill attributes names
  608. *
  609. * @param {string} content - Part of a file to attempt to parse
  610. * @since 2.15.0
  611. */
  612. tryParseAndFillAttributeNames: function (content) {
  613. var names = Utilities.tryParseCSVHeader(content);
  614. if (names.length === 0) {
  615. this.updateFillButton(
  616. '<i class="icon-warning-sign"></i> Couldn\'t fill',
  617. );
  618. } else {
  619. this.updateFillButton('<i class="icon-ok"></i> Filled!');
  620. }
  621. //Make sure the button is enabled
  622. this.enableFillButton();
  623. this.updateAttributeNames(names);
  624. },
  625. /**
  626. * Update attribute names from an array
  627. *
  628. * This will update existing attributes' names or create new
  629. * attributes as needed. This also performs a full re-render.
  630. *
  631. * @param {string[]} names - A list of names to apply
  632. * @since 2.15.0
  633. */
  634. updateAttributeNames: function (names) {
  635. if (!names) {
  636. return;
  637. }
  638. var attributes = this.model.get("attributeList");
  639. //Update the name of each attribute or create a new Attribute if one doesn't exist
  640. for (var i = 0; i < names.length; i++) {
  641. if (attributes.length - 1 >= i) {
  642. attributes[i].set("attributeName", names[i]);
  643. } else {
  644. attributes.push(
  645. new EMLAttribute({
  646. parentModel: this.model,
  647. xmlID: DataONEObject.generateId(),
  648. attributeName: names[i],
  649. }),
  650. );
  651. }
  652. }
  653. //Update the attribute list
  654. this.model.set("attributeList", attributes);
  655. // Reset first
  656. this.$(".attribute-menu.side-nav-items").empty();
  657. this.$(".eml-attribute").remove();
  658. // Then re-render
  659. this.renderAttributes();
  660. },
  661. /**
  662. * Update the Fill button temporarily and set it back to the default
  663. *
  664. * Used to show success or failure of the filling operation
  665. *
  666. * @param {string} messageHTML - HTML template string to set
  667. * temporarily
  668. * @param {boolean} disableTimeout - If true, the timeout will not be set
  669. * @since 2.15.0
  670. */
  671. updateFillButton: function (messageHTML, disableTimeout) {
  672. var view = this;
  673. this.$(".fill-button").html(messageHTML);
  674. if (!disableTimeout) {
  675. window.setTimeout(function () {
  676. view
  677. .$(".fill-button-container")
  678. .html(view.fillButtonTemplateString);
  679. }, 3000);
  680. }
  681. },
  682. /**
  683. * Disable the Fill Attributes button
  684. * @since 2.15.0
  685. */
  686. disableFillButton: function () {
  687. this.$(".fill-button").prop("disabled", true);
  688. },
  689. /**
  690. * Enable the Fill Attributes button
  691. * @since 2.15.0
  692. */
  693. enableFillButton: function () {
  694. this.$(".fill-button").prop("disabled", false);
  695. },
  696. },
  697. );
  698. return EMLEntityView;
  699. });