field_group-8.x-3.1/formatters/tabs/horizontal-tabs.js

formatters/tabs/horizontal-tabs.js
/**
 * @file
 * Provides horizontal tabs logic.
 */

(($, Drupal) => {
  Drupal.FieldGroup = Drupal.FieldGroup || {};
  Drupal.FieldGroup.Effects = Drupal.FieldGroup.Effects || {};

  /**
   * Show the parent horizontal tab pane of a targeted page fragment.
   *
   * This is actually the same as vertical-tabs.js in Drupal core.
   *
   * @param {jQuery.Event} e
   *   The event triggered.
   * @param {jQuery} $target
   *   The targeted node as a jQuery object.
   */
  const handleFragmentLinkClickOrHashChange = (e, $target) => {
    $target.parents('details.horizontal-tab-hidden').each((index, pane) => {
      $(pane).data('horizontalTab').focus();
    });
  };

  /**
   * Transforms a set of fieldsets into a stack of horizontal tabs.
   *
   * Each tab may have a summary which can be updated by another
   * script. For that to work, each fieldset has an associated
   * 'horizontalTabCallback' (with jQuery.data() attached to the fieldset),
   * which is called every time the user performs an update to a form
   * element inside the tab pane.
   */
  Drupal.behaviors.horizontalTabs = {
    attach(context) {
      const width = drupalSettings.widthBreakpoint || 640;
      const mq = `(max-width: ${width}px)`;

      /**
       * Binds a listener to handle fragment link clicks and URL hash changes.
       */
      $(once('field-group-horizontal-tabs-fragments', 'body')).on(
        'formFragmentLinkClickOrHashChange.horizontalTabs',
        handleFragmentLinkClickOrHashChange,
      );

      if (window.matchMedia(mq).matches) {
        return;
      }

      $(once('horizontal-tabs', '[data-horizontal-tabs]', context)).each(
        (indexTabsWrapper, tabsWrapper) => {
          const horizontalTabsClearfix = tabsWrapper;
          $(tabsWrapper)
            .find('> [data-horizontal-tabs-panes]')
            .each((indexTabWrapper, tabWrapper) => {
              const $this = $(tabWrapper).addClass('horizontal-tabs-panes');
              const focusID = $(
                ':hidden.horizontal-tabs-active-tab',
                tabWrapper,
              ).val();
              let tabFocus;

              // Check if there are some details that can be converted to horizontal-tabs.
              const $details = $this.find('> details');
              if ($details.length === 0) {
                return;
              }

              // Find the tab column.
              const tabList = $(horizontalTabsClearfix).find(
                '> [data-horizontal-tabs-list]',
              );
              tabList.removeClass('visually-hidden');

              let tabTitle;
              /* eslint max-nested-callbacks: ['error', 4] */
              // Transform each details into a tab.
              $details.each((i, element) => {
                const $thisDetail = $(element);
                const $summaryElement = $thisDetail.find('> summary');
                const $detailsTitle = $summaryElement
                  .first()
                  .find('.details-title');
                if ($detailsTitle.length) {
                  tabTitle = $detailsTitle.find('> span:last-child').html();
                } else {
                  // Remove the required mark element from the tab title, if it
                  // exists (e.g. gin theme will have it twice).
                  tabTitle =
                    $summaryElement
                      .clone()
                      .find('.required-mark')
                      .remove()
                      .end()
                      .html() ||
                    $summaryElement.find('> span:first-child').html();
                }

                const horizontalTab = new Drupal.HorizontalTab({
                  title: tabTitle,
                  details: $thisDetail,
                });
                horizontalTab.item.addClass(`horizontal-tab-button-${i}`);
                horizontalTab.item.attr('data-horizontalTabButton', i);
                tabList.append(horizontalTab.item);
                $thisDetail
                  .removeClass('collapsed')
                  // prop() can't be used on browsers not supporting details element,
                  // the style won't apply to them if prop() is used.
                  .attr('open', true)
                  .addClass('horizontal-tabs-pane')
                  .data('horizontalTab', horizontalTab);
                if (element.id === focusID) {
                  tabFocus = $thisDetail;
                }
              });

              $(tabList).find('> li:first').addClass('first');
              $(tabList).find('> li:last').addClass('last');

              if (!tabFocus) {
                // If the current URL has a fragment and one of the tabs contains an
                // element that matches the URL fragment, activate that tab.
                // eslint-disable-next-line no-useless-escape
                const hash = window.location.hash.replace(/[=%;,\/]/g, '');
                if (hash !== '#' && $(hash, this).length) {
                  tabFocus = $(hash, this).closest('.horizontal-tabs-pane');
                } else {
                  tabFocus = $this.find('> .horizontal-tabs-pane:first');
                }
              }
              if (tabFocus.length) {
                tabFocus.data('horizontalTab').focus();
              }
            });
        },
      );
    },
  };

  /**
   * The horizontal tab object represents a single tab within a tab group.
   *
   * @param {object} settings
   *   An object with the following keys:
   *   - title: The name of the tab.
   *   - details: The jQuery object of the details element that is the tab pane.
   */
  Drupal.HorizontalTab = function horizontalTab(settings) {
    const self = this;
    $.extend(this, settings, Drupal.theme('horizontalTab', settings));

    this.link.attr('href', `#${settings.details.attr('id')}`);

    this.link.on('click', (e) => {
      e.preventDefault();
      self.focus();
    });

    // Keyboard events added:
    // Pressing the Enter key will open the tab pane.
    this.link.on('keydown', (event) => {
      if (event.keyCode === 13) {
        event.preventDefault();
        self.focus();
        // Set focus on the first input field of the visible details/tab pane.
        $('.horizontal-tabs-pane :input:visible:enabled:first').trigger(
          'focus',
        );
      }
    });

    // Only bind update summary on forms.
    if (this.details.drupalGetSummary) {
      this.details
        .on('summaryUpdated', () => {
          self.updateSummary();
        })
        .trigger('summaryUpdated');
    }
  };

  Drupal.HorizontalTab.prototype = {
    /**
     * Displays the tab's content pane.
     */
    focus() {
      this.details
        .removeClass('horizontal-tab-hidden')
        .siblings('.horizontal-tabs-pane')
        .each((index, element) => {
          const tab = $(element).data('horizontalTab');
          tab.details.addClass('horizontal-tab-hidden');
          tab.details.hide();
          tab.item.removeClass('selected');
        })
        .end()
        .show()
        .siblings(':hidden.horizontal-tabs-active-tab')
        .val(this.details.attr('id'));
      this.item.addClass('selected');
      // Mark the active tab for screen readers.
      $('#active-horizontal-tab').remove();
      this.link.append(
        `<span id="active-horizontal-tab" class="visually-hidden">${Drupal.t(
          '(active tab)',
        )}</span>`,
      );
    },

    /**
     * Updates the tab's summary.
     */
    updateSummary() {
      this.summary.html(this.details.drupalGetSummary());
    },

    /**
     * Shows a horizontal tab pane.
     *
     * @return {Drupal.HorizontalTab} The current horizontal tab.
     */
    tabShow() {
      // Display the tab.
      this.item.removeClass('horizontal-tab-hidden');
      this.item.show();

      // Update .first marker for items. We need recurse from parent to retain the
      // actual DOM element order as jQuery implements sortOrder, but not as public
      // method.
      this.item
        .parent()
        .children('.horizontal-tab-button')
        .removeClass('first')
        .filter(':visible:first')
        .addClass('first');
      // Display the details element.
      this.details.removeClass('horizontal-tab-hidden');
      // Focus this tab.
      this.focus();
      return this;
    },

    /**
     * Hides a horizontal tab pane.
     *
     * @return {Drupal.HorizontalTab} The current horizontal tab.
     */
    tabHide() {
      // Hide this tab.
      this.item.addClass('horizontal-tab-hidden');
      this.item.hide();

      // Update .first marker for items. We need recurse from parent to retain the
      // actual DOM element order as jQuery implements sortOrder, but not as public
      // method.
      this.item
        .parent()
        .children('.horizontal-tab-button')
        .removeClass('first')
        .filter(':visible:first')
        .addClass('first');
      // Hide the details element.
      this.details.addClass('horizontal-tab-hidden');
      // Focus the first visible tab (if there is one).
      const $firstTab = this.details.siblings(
        '.horizontal-tabs-pane:not(.horizontal-tab-hidden):first',
      );
      if ($firstTab.length) {
        $firstTab.data('horizontalTab').focus();
      } else {
        // Hide the vertical tabs (if no tabs remain).
        this.item.closest('.form-type-horizontal-tabs').hide();
      }
      return this;
    },
  };

  /**
   * Theme function for a horizontal tab.
   *
   * @param {object} settings
   *   An object with the following keys:
   *   - title: The name of the tab.
   *
   * @return {object}
   *   This function has to return an object with at least these keys:
   *   - item: The root tab jQuery element
   *   - link: The anchor tag that acts as the clickable area of the tab
   *       (jQuery version)
   *   - summary: The jQuery element that contains the tab summary
   */
  Drupal.theme.horizontalTab = (settings) => {
    const tab = {};
    const idAttr = settings.details.attr('id');

    tab.item = $(
      '<li class="horizontal-tab-button" tabindex="-1"></li>',
    ).append(
      (tab.link = $(`<a href="#${idAttr}"></a>`).append(
        (tab.title = $('<strong></strong>').html(settings.title)),
      )),
    );

    tab.item.addClass(settings.details.attr('class'));
    // Remove 'claro-details' class to remove additional bottom border from ul >
    // li of horizontal tab.
    tab.item.removeClass('claro-details');

    // No need to add summary on frontend.
    if (settings.details.drupalGetSummary) {
      tab.link.append((tab.summary = $('<span class="summary"></span>')));
    }

    return tab;
  };
})(jQuery, Drupal);

Главная | Обратная связь

drupal hosting | друпал хостинг | it patrol .inc