Source: src/js/views/accordion/AccordionView.js

  1. define([
  2. "jquery",
  3. "backbone",
  4. "semantic",
  5. "views/accordion/AccordionItemView",
  6. "models/accordion/Accordion",
  7. ], ($, Backbone, Semantic, AccordionItemView, AccordionModel) => {
  8. // The base class for the view
  9. const BASE_CLASS = "accordion-view";
  10. /**
  11. * @class AccordionView
  12. * @classdesc An extension of the Semantic UI accordion that allows for
  13. * defining contents with a Backbone model, and adds tooltips and other
  14. * features.
  15. * @classcategory Views/Accordion
  16. * @augments Backbone.View
  17. * @class
  18. * @since 2.31.0
  19. * @screenshot views/accordion/AccordionView.png
  20. */
  21. const AccordionView = Backbone.View.extend(
  22. /** @lends AccordionView.prototype */
  23. {
  24. /** @inheritdoc */
  25. type: "AccordionView",
  26. /** @inheritdoc */
  27. className: BASE_CLASS,
  28. /** @inheritdoc */
  29. tagName: "div",
  30. /**
  31. * Initializes the AccordionView with the model and listens for changes
  32. * to the items collection.
  33. * @param {object} options - Options for the view
  34. * @param {Accordion} [options.model] - The Accordion model for the view. If
  35. * not provided, a new Accordion model will be created.
  36. * @param {object} [options.modelData] - Optional data to initialize the
  37. * Accordion model with. Only used if options.model is not provided.
  38. */
  39. initialize(options) {
  40. // Set the model to the provided model or create a new one
  41. this.model = options?.model || new AccordionModel(options?.modelData);
  42. this.listenTo(this.model.get("items"), "add", this.addNewItem);
  43. this.listenTo(this.model.get("items"), "remove", this.removeItem);
  44. },
  45. /** @inheritdoc */
  46. render() {
  47. this.initializeAccordion();
  48. // Start rendering the root, and then the children will be rendered
  49. // recursively
  50. const rootItems = this.model.getRootItems();
  51. const rootAccordion = this.createAccordion(rootItems);
  52. this.rootAccordion = rootAccordion;
  53. this.$el.append(rootAccordion);
  54. return this;
  55. },
  56. /**
  57. * Initializes the Semantic UI accordion module with the settings from the
  58. * model. The module handles applying accordion behavior to the view and
  59. * any new DOM elements that are added.
  60. */
  61. initializeAccordion() {
  62. const view = this;
  63. // Gather all the necessary settings from the model
  64. const modelJSON = this.model.toJSON();
  65. const settings = Object.fromEntries(
  66. Object.entries(modelJSON).filter(([key, _]) =>
  67. Semantic.ACCORDION_SETTINGS_KEYS.includes(key),
  68. ),
  69. );
  70. // Pass the associated model and view to the callback functions instead
  71. // only having access to the DOM element as the context
  72. const createCallback = (callbackName) => {
  73. const originalCallback = this.model.get(callbackName);
  74. if (originalCallback) {
  75. return function callbackWrapper() {
  76. const contentEl = this[0];
  77. const itemView = view.viewFromContentEl(contentEl);
  78. const itemModel = itemView.model;
  79. originalCallback(itemModel, itemView);
  80. };
  81. }
  82. return null;
  83. };
  84. Semantic.ACCORDION_CALLBACKS.forEach((callbackName) => {
  85. const callback = createCallback(callbackName);
  86. if (callback) {
  87. settings[callbackName] = callback;
  88. } else {
  89. delete settings[callbackName];
  90. }
  91. });
  92. // Initialize the accordion with the specified settings
  93. this.$el.accordion(settings);
  94. },
  95. /**
  96. * @param {HTMLElement} contentEl A content element from an item
  97. * @returns {AccordionItemView} The view associated with the item
  98. */
  99. viewFromContentEl(contentEl) {
  100. const views = Object.values(this.itemViews);
  101. return views.find((view) => view.contentContainer === contentEl);
  102. },
  103. /**
  104. * Creates a container for the accordion with the necessary classes for
  105. * Semantic UI and renders any items belonging to the accordion. This can
  106. * be used to create the root accordion or nested accordions.
  107. * @param {AccordionItem[]} items - An array of AccordionItem models
  108. * @returns {HTMLElement} The container element for the accordion
  109. */
  110. createAccordion(items) {
  111. const accordionContainer = this.createContainer();
  112. if (items?.length) this.addItems(items, accordionContainer);
  113. return accordionContainer;
  114. },
  115. /**
  116. * Creates a container element for the accordion with the necessary
  117. * classes
  118. * @returns {HTMLElement} The container element for the accordion
  119. */
  120. createContainer() {
  121. const container = document.createElement("div");
  122. container.classList.add(
  123. Semantic.CLASS_NAMES.accordion.container,
  124. Semantic.CLASS_NAMES.base,
  125. );
  126. container.style.marginTop = 0;
  127. // Add optional class names based on model properties
  128. const optionalClasses = ["fluid", "styled", "inverted"];
  129. optionalClasses.forEach((className) => {
  130. if (this.model.get(className)) {
  131. container.classList.add(Semantic.CLASS_NAMES.variations[className]);
  132. }
  133. });
  134. return container;
  135. },
  136. /**
  137. * Adds items to the container element for the accordion
  138. * @param {AccordionItem[]} models - An array of AccordionItem models
  139. * @param {HTMLElement} container - The container element for the
  140. * accordion
  141. */
  142. addItems(models, container) {
  143. models.forEach((model) => {
  144. this.addItem(model, container);
  145. });
  146. },
  147. /**
  148. * Adds an item to the container element for the accordion
  149. * @param {AccordionItem} model - An AccordionItem model
  150. * @param {HTMLElement} container - The container element for the
  151. * accordion
  152. */
  153. addItem(model, container) {
  154. const itemView = new AccordionItemView({ model }).render();
  155. // Semantic UI expects the title and content to be direct children of
  156. // the accordion container, important for correct application of CSS
  157. container.appendChild(itemView.titleContainer);
  158. container.appendChild(itemView.contentContainer);
  159. if (!this.itemViews) this.itemViews = {};
  160. const id = model.get("itemId");
  161. this.itemViews[id] = itemView;
  162. // Add children if they exist, otherwise just re-adds the content
  163. this.refreshContent(id);
  164. },
  165. /**
  166. * Refreshes the content of an item in the accordion. If the item has
  167. * children, it will create a new accordion with the children. Otherwise,
  168. * it will update the content with the item's model content attribute.
  169. * @param {string} itemId - The model itemId for the item
  170. */
  171. refreshContent(itemId) {
  172. // Check if there are children for the given item
  173. const children = this.model.getChildren(itemId);
  174. const itemView = this.itemViews?.[itemId];
  175. if (!itemView) return;
  176. if (children?.length) {
  177. const subAccordion = this.createAccordion(children);
  178. itemView.updateContent(subAccordion);
  179. } else {
  180. itemView.updateContent(itemView.model.get("content"));
  181. }
  182. },
  183. /**
  184. * Handles adding a new item to the accordion when the items collection
  185. * has new models added to it.
  186. * @param {AccordionItem} model - The new AccordionItem model
  187. */
  188. addNewItem(model) {
  189. const parent = model.get("parent");
  190. if (parent) {
  191. this.refreshContent(parent);
  192. } else {
  193. this.addItem(model, this.rootAccordion);
  194. }
  195. },
  196. /**
  197. * Handles removing an item from the accordion when the items collection
  198. * has models removed from it.
  199. * @param {AccordionItem} model - The removed AccordionItem model
  200. */
  201. removeItem(model) {
  202. const id = model.get("itemId") || model.cid;
  203. const itemView = this.itemViews[id];
  204. // remove the item view from the itemViews object
  205. delete this.itemViews[id];
  206. itemView?.remove();
  207. },
  208. /**
  209. * Removes all items from the accordion and clears the itemViews object.
  210. */
  211. clearAllItems() {
  212. Object.entries(this.itemViews).forEach(([id, view]) => {
  213. view.remove();
  214. delete this.itemViews[id];
  215. });
  216. },
  217. },
  218. );
  219. return AccordionView;
  220. });