FileMaster
Search
Toggle Dark Mode
Home
/
.
/
wp-content
/
plugins
/
booking
/
_dist
/
all
/
_src
Edit File: wpbc_all_admin.js
/** * Blink specific HTML element to set attention to this element. * * @param {string} element_to_blink - class or id of element: '.wpbc_widget_available_unavailable' * @param {int} how_many_times - 4 * @param {int} how_long_to_blink - 350 */ function wpbc_blink_element( element_to_blink, how_many_times = 4, how_long_to_blink = 350 ){ for ( let i = 0; i < how_many_times; i++ ){ jQuery( element_to_blink ).fadeOut( how_long_to_blink ).fadeIn( how_long_to_blink ); } jQuery( element_to_blink ).animate( {opacity: 1}, 500 ); } /** * Support Functions - Spin Icon in Buttons ------------------------------------------------------------------ */ /** * Remove spin icon from button and Enable this button. * * @param button_clicked_element_id - HTML ID attribute of this button * @return string - CSS classes that was previously in button icon */ function wpbc_button__remove_spin(button_clicked_element_id) { var previos_classes = ''; if ( (undefined != button_clicked_element_id) && ('' != button_clicked_element_id) ) { var jElement = jQuery( '#' + button_clicked_element_id ); if ( jElement.length ) { previos_classes = wpbc_button_disable_loading_icon( jElement.get( 0 ) ); } } return previos_classes; } /** * Show Loading (rotating arrow) icon for button that has been clicked * * @param this_button - this object of specific button * @return string - CSS classes that was previously in button icon */ function wpbc_button_enable_loading_icon(this_button) { var jButton = jQuery( this_button ); var jIcon = jButton.find( 'i' ); var previos_classes = jIcon.attr( 'class' ); jIcon.removeClass().addClass( 'menu_icon icon-1x wpbc_icn_rotate_right wpbc_spin' ); // Set Rotate icon. // jIcon.addClass( 'wpbc_animation_pause' ); // Pause animation. // jIcon.addClass( 'wpbc_ui_red' ); // Set icon color red. jIcon.attr( 'wpbc_previous_class', previos_classes ) jButton.addClass( 'disabled' ); // Disable button // We need to set here attr instead of prop, because for A elements, attribute 'disabled' do not added with jButton.prop( "disabled", true );. jButton.attr( 'wpbc_previous_onclick', jButton.attr( 'onclick' ) ); // Save this value. jButton.attr( 'onclick', '' ); // Disable actions "on click". return previos_classes; } /** * Hide Loading (rotating arrow) icon for button that was clicked and show previous icon and enable button * * @param this_button - this object of specific button * @return string - CSS classes that was previously in button icon */ function wpbc_button_disable_loading_icon(this_button) { var jButton = jQuery( this_button ); var jIcon = jButton.find( 'i' ); var previos_classes = jIcon.attr( 'wpbc_previous_class' ); if ( (undefined != previos_classes) && ('' != previos_classes) ) { jIcon.removeClass().addClass( previos_classes ); } jButton.removeClass( 'disabled' ); // Remove Disable button. var previous_onclick = jButton.attr( 'wpbc_previous_onclick' ) if ( (undefined != previous_onclick) && ('' != previous_onclick) ) { jButton.attr( 'onclick', previous_onclick ); } return previos_classes; } /** * On selection of radio button, adjust attributes of radio container * * @param _this */ function wpbc_ui_el__radio_container_selection(_this) { if ( jQuery( _this ).is( ':checked' ) ) { jQuery( _this ).parents( '.wpbc_ui_radio_section' ).find( '.wpbc_ui_radio_container' ).removeAttr( 'data-selected' ); jQuery( _this ).parents( '.wpbc_ui_radio_container:not(.disabled)' ).attr( 'data-selected', true ); } if ( jQuery( _this ).is( ':disabled' ) ) { jQuery( _this ).parents( '.wpbc_ui_radio_container' ).addClass( 'disabled' ); } } /** * On click on Radio Container, we will select the radio button and then adjust attributes of radio container * * @param _this */ function wpbc_ui_el__radio_container_click(_this) { if ( jQuery( _this ).hasClass( 'disabled' ) ) { return false; } var j_radio = jQuery( _this ).find( 'input[type=radio]:not(.wpbc-form-radio-internal)' ); if ( j_radio.length ) { j_radio.prop( 'checked', true ).trigger( 'change' ); } } "use strict"; // ===================================================================================================================== // == Full Screen - support functions == // ===================================================================================================================== /** * Check Full screen mode, by removing top tab */ function wpbc_check_full_screen_mode(){ if ( jQuery( 'body' ).hasClass( 'wpbc_admin_full_screen' ) ) { jQuery( 'html' ).removeClass( 'wp-toolbar' ); } else { jQuery( 'html' ).addClass( 'wp-toolbar' ); } wpbc_check_buttons_max_min_in_full_screen_mode(); } function wpbc_check_buttons_max_min_in_full_screen_mode() { if ( jQuery( 'body' ).hasClass( 'wpbc_admin_full_screen' ) ) { jQuery( '.wpbc_ui__top_nav__btn_full_screen' ).addClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_normal_screen' ).removeClass( 'wpbc_ui__hide' ); } else { jQuery( '.wpbc_ui__top_nav__btn_full_screen' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_normal_screen' ).addClass( 'wpbc_ui__hide' ); } } jQuery( document ).ready( function () { wpbc_check_full_screen_mode(); } ); /** * Checkbox Selection functions for Listing. */ /** * Selections of several checkboxes like in gMail with shift :) * Need to have this structure: * .wpbc_selectable_table * .wpbc_selectable_head * .check-column * :checkbox * .wpbc_selectable_body * .wpbc_row * .check-column * :checkbox * .wpbc_selectable_foot * .check-column * :checkbox */ function wpbc_define_gmail_checkbox_selection( $ ){ var checks, first, last, checked, sliced, lastClicked = false; // Check all checkboxes. $( '.wpbc_selectable_body' ).find( '.check-column' ).find( ':checkbox' ).on( 'click', function (e) { if ( 'undefined' == e.shiftKey ) { return true; } if ( e.shiftKey ) { if ( ! lastClicked ) { return true; } checks = $( lastClicked ).closest( '.wpbc_selectable_body' ).find( ':checkbox' ).filter( ':visible:enabled' ); first = checks.index( lastClicked ); last = checks.index( this ); checked = $( this ).prop( 'checked' ); if ( 0 < first && 0 < last && first != last ) { sliced = (last > first) ? checks.slice( first, last ) : checks.slice( last, first ); sliced.prop( 'checked', function () { if ( $( this ).closest( '.wpbc_row' ).is( ':visible' ) ) { return checked; } return false; } ).trigger( 'change' ); } } lastClicked = this; // toggle "check all" checkboxes. var unchecked = $( this ).closest( '.wpbc_selectable_body' ).find( ':checkbox' ).filter( ':visible:enabled' ).not( ':checked' ); $( this ).closest( '.wpbc_selectable_table' ).children( '.wpbc_selectable_head, .wpbc_selectable_foot' ).find( ':checkbox' ).prop( 'checked', function () { return (0 === unchecked.length); } ).trigger( 'change' ); return true; } ); // Head || Foot clicking to select / deselect ALL. $( '.wpbc_selectable_head, .wpbc_selectable_foot' ).find( '.check-column :checkbox' ).on( 'click', function (event) { var $this = $( this ), $table = $this.closest( '.wpbc_selectable_table' ), controlChecked = $this.prop( 'checked' ), toggle = event.shiftKey || $this.data( 'wp-toggle' ); $table.children( '.wpbc_selectable_body' ).filter( ':visible' ) .find( '.check-column' ).find( ':checkbox' ) .prop( 'checked', function () { if ( $( this ).is( ':hidden,:disabled' ) ) { return false; } if ( toggle ) { return ! $( this ).prop( 'checked' ); } else if ( controlChecked ) { return true; } return false; } ).trigger( 'change' ); $table.children( '.wpbc_selectable_head, .wpbc_selectable_foot' ).filter( ':visible' ) .find( '.check-column' ).find( ':checkbox' ) .prop( 'checked', function () { if ( toggle ) { return false; } else if ( controlChecked ) { return true; } return false; } ); } ); // Visually show selected border. $( '.wpbc_selectable_body' ).find( '.check-column :checkbox' ).on( 'change', function (event) { if ( jQuery( this ).is( ':checked' ) ) { jQuery( this ).closest( '.wpbc_list_row' ).addClass( 'row_selected_color' ); } else { jQuery( this ).closest( '.wpbc_list_row' ).removeClass( 'row_selected_color' ); } // Disable text selection while pressing 'shift'. document.getSelection().removeAllRanges(); // Show or hide buttons on Actions toolbar at Booking Listing page, if we have some selected bookings. wpbc_show_hide_action_buttons_for_selected_bookings(); } ); wpbc_show_hide_action_buttons_for_selected_bookings(); } /** * Get ID array of selected elements */ function wpbc_get_selected_row_id() { var $table = jQuery( '.wpbc__wrap__booking_listing .wpbc_selectable_table' ); var checkboxes = $table.children( '.wpbc_selectable_body' ).filter( ':visible' ).find( '.check-column' ).find( ':checkbox' ); var selected_id = []; jQuery.each( checkboxes, function (key, checkbox) { if ( jQuery( checkbox ).is( ':checked' ) ) { var element_id = wpbc_get_row_id_from_element( checkbox ); selected_id.push( element_id ); } } ); return selected_id; } /** * Get ID of row, based on clciked element * * @param this_inbound_element - ususlly this * @returns {number} */ function wpbc_get_row_id_from_element(this_inbound_element) { var element_id = jQuery( this_inbound_element ).closest( '.wpbc_listing_usual_row' ).attr( 'id' ); element_id = parseInt( element_id.replace( 'row_id_', '' ) ); return element_id; } /** * == Booking Listing == Show or hide buttons on Actions toolbar at page, if we have some selected bookings. */ function wpbc_show_hide_action_buttons_for_selected_bookings(){ var selected_rows_arr = wpbc_get_selected_row_id(); if ( selected_rows_arr.length > 0 ) { jQuery( '.hide_button_if_no_selection' ).show(); } else { jQuery( '.hide_button_if_no_selection' ).hide(); } } "use strict"; // ===================================================================================================================== // == Left Bar - expand / colapse functions == // ===================================================================================================================== /** * Expand Vertical Left Bar. */ function wpbc_admin_ui__sidebar_left__do_max() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'max' ); jQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).addClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' ); jQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_max' ); } /** * Hide Vertical Left Bar. */ function wpbc_admin_ui__sidebar_left__do_min() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'min' ); jQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).addClass( 'wpbc_ui__hide' ); jQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' ); jQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_min' ); } /** * Colapse Vertical Left Bar. */ function wpbc_admin_ui__sidebar_left__do_compact() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'compact' ); jQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).addClass( 'wpbc_ui__hide' ); jQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' ); jQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_compact' ); } /** * Completely Hide Vertical Left Bar. */ function wpbc_admin_ui__sidebar_left__do_hide() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'none' ); jQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).addClass( 'wpbc_ui__hide' ); // Hide top "Menu" button with divider. jQuery( '.wpbc_ui__top_nav__btn_show_left_vertical_nav,.wpbc_ui__top_nav__btn_show_left_vertical_nav_divider' ).addClass( 'wpbc_ui__hide' ); jQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' ); jQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_none' ); } /** * Action on click "Go Back" - show root menu * or some other section in left sidebar. * * @param string menu_to_show - menu slug. */ function wpbc_admin_ui__sidebar_left__show_section( menu_to_show ) { jQuery( '.wpbc_ui_el__vert_left_bar__section' ).addClass( 'wpbc_ui__hide' ) jQuery( '.wpbc_ui_el__vert_left_bar__section_' + menu_to_show ).removeClass( 'wpbc_ui__hide' ); } // ===================================================================================================================== // == Right Side Bar - expand / colapse functions == // ===================================================================================================================== /** * Expand Vertical Right Bar. */ function wpbc_admin_ui__sidebar_right__do_max() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'max_right' ); jQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).addClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); } /** * Hide Vertical Right Bar. */ function wpbc_admin_ui__sidebar_right__do_min() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'min_right' ); jQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).addClass( 'wpbc_ui__hide' ); } /** * Colapse Vertical Right Bar. */ function wpbc_admin_ui__sidebar_right__do_compact() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'compact_right' ); jQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).addClass( 'wpbc_ui__hide' ); } /** * Completely Hide Vertical Right Bar. */ function wpbc_admin_ui__sidebar_right__do_hide() { jQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' ); jQuery( '.wpbc_settings_page_wrapper' ).addClass( 'none_right' ); jQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' ); jQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).addClass( 'wpbc_ui__hide' ); // Hide top "Menu" button with divider. jQuery( '.wpbc_ui__top_nav__btn_show_right_vertical_nav,.wpbc_ui__top_nav__btn_show_right_vertical_nav_divider' ).addClass( 'wpbc_ui__hide' ); } /** * Action on click "Go Back" - show root menu * or some other section in right sidebar. * * @param string menu_to_show - menu slug. */ function wpbc_admin_ui__sidebar_right__show_section( menu_to_show ) { jQuery( '.wpbc_ui_el__vert_right_bar__section' ).addClass( 'wpbc_ui__hide' ) jQuery( '.wpbc_ui_el__vert_right_bar__section_' + menu_to_show ).removeClass( 'wpbc_ui__hide' ); } // ===================================================================================================================== // == End Right Side Bar section == // ===================================================================================================================== /** * Get anchor(s) array from URL. * Doc: https://developer.mozilla.org/en-US/docs/Web/API/Location * * @returns {*[]} */ function wpbc_url_get_anchors_arr() { var hashes = window.location.hash.replace( '%23', '#' ); var hashes_arr = hashes.split( '#' ); var result = []; var hashes_arr_length = hashes_arr.length; for ( var i = 0; i < hashes_arr_length; i++ ) { if ( hashes_arr[i].length > 0 ) { result.push( hashes_arr[i] ); } } return result; } /** * Auto Expand Settings section based on URL anchor, after page loaded. */ jQuery( document ).ready( function () { wpbc_admin_ui__do_expand_section(); setTimeout( 'wpbc_admin_ui__do_expand_section', 10 ); } ); jQuery( document ).ready( function () { wpbc_admin_ui__do_expand_section(); setTimeout( 'wpbc_admin_ui__do_expand_section', 150 ); } ); /** * Expand section in General Settings page and select Menu item. */ function wpbc_admin_ui__do_expand_section() { // window.location.hash = #section_id / doc: https://developer.mozilla.org/en-US/docs/Web/API/Location . var anchors_arr = wpbc_url_get_anchors_arr(); var anchors_arr_length = anchors_arr.length; if ( anchors_arr_length > 0 ) { var one_anchor_prop_value = anchors_arr[0].split( 'do_expand__' ); if ( one_anchor_prop_value.length > 1 ) { // 'wpbc_general_settings_calendar_metabox' var section_to_show = one_anchor_prop_value[1]; var section_id_to_show = '#' + section_to_show; // -- Remove selected background in all left menu items --------------------------------------------------- jQuery( '.wpbc_ui_el__vert_nav_item ' ).removeClass( 'active' ); // Set left menu selected. jQuery( '.do_expand__' + section_to_show + '_link' ).addClass( 'active' ); var selected_title = jQuery( '.do_expand__' + section_to_show + '_link a .wpbc_ui_el__vert_nav_title ' ).text(); // Expand section, if it colapsed. if ( ! jQuery( '.do_expand__' + section_to_show + '_link' ).parents( '.wpbc_ui_el__level__folder' ).hasClass( 'expanded' ) ) { jQuery( '.wpbc_ui_el__level__folder' ).removeClass( 'expanded' ); jQuery( '.do_expand__' + section_to_show + '_link' ).parents( '.wpbc_ui_el__level__folder' ).addClass( 'expanded' ); } // -- Expand section --------------------------------------------------------------------------------------- var container_to_hide_class = '.postbox'; // Hide sections '.postbox' in admin page and show specific one. jQuery( '.wpbc_admin_page ' + container_to_hide_class ).hide(); jQuery( '.wpbc_container_always_hide__on_left_nav_click' ).hide(); jQuery( section_id_to_show ).show(); // Show all other sections, if provided in URL: ..?page=wpbc-settings#do_expand__wpbc_general_settings_capacity_metabox#wpbc_general_settings_capacity_upgrade_metabox . for ( let i = 1; i < anchors_arr_length; i++ ) { jQuery( '#' + anchors_arr[i] ).show(); } if ( false ) { var targetOffset = wpbc_scroll_to( section_id_to_show ); } // -- Set Value to Input about selected Nav element --------------------------------------------------------------- // FixIn: 9.8.6.1. var section_id_tab = section_id_to_show.substring( 0, section_id_to_show.length - 8 ) + '_tab'; if ( container_to_hide_class == section_id_to_show ) { section_id_tab = '#wpbc_general_settings_all_tab' } if ( '#wpbc_general_settings_capacity_metabox,#wpbc_general_settings_capacity_upgrade_metabox' == section_id_to_show ) { section_id_tab = '#wpbc_general_settings_capacity_tab' } jQuery( '#form_visible_section' ).val( section_id_tab ); } // Like blinking some elements. wpbc_admin_ui__do__anchor__another_actions(); } } function wpbc_admin_ui__is_in_mobile_screen_size() { return wpbc_admin_ui__is_in_this_screen_size( 605 ); } function wpbc_admin_ui__is_in_this_screen_size(size) { return (window.screen.width <= size); } /** * Open settings page | Expand section | Select Menu item. */ function wpbc_admin_ui__do__open_url__expand_section(url, section_id) { // window.location.href = url + '&do_expand=' + section_id + '#do_expand__' + section_id; //. window.location.href = url + '#do_expand__' + section_id; if ( wpbc_admin_ui__is_in_mobile_screen_size() ) { wpbc_admin_ui__sidebar_left__do_min(); } wpbc_admin_ui__do_expand_section(); } /** * Check for Other actions: Like blinking some elements in settings page. E.g. Days selection or change-over days. */ function wpbc_admin_ui__do__anchor__another_actions() { var anchors_arr = wpbc_url_get_anchors_arr(); var anchors_arr_length = anchors_arr.length; // Other actions: Like blinking some elements. for ( var i = 0; i < anchors_arr_length; i++ ) { var this_anchor = anchors_arr[i]; var this_anchor_prop_value = this_anchor.split( 'do_other_actions__' ); if ( this_anchor_prop_value.length > 1 ) { var section_action = this_anchor_prop_value[1]; switch ( section_action ) { case 'blink_day_selections': // wpbc_ui_settings__panel__click( '#wpbc_general_settings_calendar_tab a', '#wpbc_general_settings_calendar_metabox', 'Days Selection' );. wpbc_blink_element( '.wpbc_tr_set_gen_booking_type_of_day_selections', 4, 350 ); wpbc_scroll_to( '.wpbc_tr_set_gen_booking_type_of_day_selections' ); break; case 'blink_change_over_days': // wpbc_ui_settings__panel__click( '#wpbc_general_settings_calendar_tab a', '#wpbc_general_settings_calendar_metabox', 'Changeover Days' );. wpbc_blink_element( '.wpbc_tr_set_gen_booking_range_selection_time_is_active', 4, 350 ); wpbc_scroll_to( '.wpbc_tr_set_gen_booking_range_selection_time_is_active' ); break; case 'blink_captcha': wpbc_blink_element( '.wpbc_tr_set_gen_booking_is_use_captcha', 4, 350 ); wpbc_scroll_to( '.wpbc_tr_set_gen_booking_is_use_captcha' ); break; default: } } } } /** * Copy txt to clipbrd from Text fields. * * @param html_element_id - e.g. 'data_field' * @returns {boolean} */ function wpbc_copy_text_to_clipbrd_from_element( html_element_id ) { // Get the text field. var copyText = document.getElementById( html_element_id ); // Select the text field. copyText.select(); copyText.setSelectionRange( 0, 99999 ); // For mobile devices. // Copy the text inside the text field. var is_copied = wpbc_copy_text_to_clipbrd( copyText.value ); if ( ! is_copied ) { console.error( 'Oops, unable to copy', copyText.value ); } return is_copied; } /** * Copy txt to clipbrd. * * @param text * @returns {boolean} */ function wpbc_copy_text_to_clipbrd(text) { if ( ! navigator.clipboard ) { return wpbc_fallback_copy_text_to_clipbrd( text ); } navigator.clipboard.writeText( text ).then( function () { // console.log( 'Async: Copying to clipboard was successful!' );. return true; }, function (err) { // console.error( 'Async: Could not copy text: ', err );. return false; } ); } /** * Copy txt to clipbrd - depricated method. * * @param text * @returns {boolean} */ function wpbc_fallback_copy_text_to_clipbrd( text ) { // ----------------------------------------------------------------------------------------------------------------- // var textArea = document.createElement( "textarea" ); // textArea.value = text; // // // Avoid scrolling to bottom. // textArea.style.top = "0"; // textArea.style.left = "0"; // textArea.style.position = "fixed"; // textArea.style.zIndex = "999999999"; // document.body.appendChild( textArea ); // textArea.focus(); // textArea.select(); // ----------------------------------------------------------------------------------------------------------------- // Now get it as HTML (original here https://stackoverflow.com/questions/34191780/javascript-copy-string-to-clipboard-as-text-html ). // [1] - Create container for the HTML. var container = document.createElement( 'div' ); container.innerHTML = text; // [2] - Hide element. container.style.position = 'fixed'; container.style.pointerEvents = 'none'; container.style.opacity = 0; // Detect all style sheets of the page. var activeSheets = Array.prototype.slice.call( document.styleSheets ).filter( function (sheet) { return ! sheet.disabled; } ); // [3] - Mount the container to the DOM to make `contentWindow` available. document.body.appendChild( container ); // [4] - Copy to clipboard. window.getSelection().removeAllRanges(); var range = document.createRange(); range.selectNode( container ); window.getSelection().addRange( range ); // ----------------------------------------------------------------------------------------------------------------- var result = false; try { result = document.execCommand( 'copy' ); // console.log( 'Fallback: Copying text command was ' + msg ); //. } catch ( err ) { // console.error( 'Fallback: Oops, unable to copy', err ); //. } // document.body.removeChild( textArea ); //. // [5.4] - Enable CSS. var activeSheets_length = activeSheets.length; for ( var i = 0; i < activeSheets_length; i++ ) { activeSheets[i].disabled = false; } // [6] - Remove the container document.body.removeChild( container ); return result; } /** * WPBC Collapsible Groups * * Universal, dependency-free controller for expanding/collapsing grouped sections in right-side panels (Inspector/Library/Form Settings, or any other WPBC page). * * === How to use it (quick) ? === * * -- 1. Markup (independent mode: multiple open allowed) -- * <div class="wpbc_collapsible"> * <section class="wpbc_ui__collapsible_group is-open"> * <button type="button" class="group__header"><h3>General</h3></button> * <div class="group__fields">…</div> * </section> * <section class="wpbc_ui__collapsible_group"> * <button type="button" class="group__header"><h3>Advanced</h3></button> * <div class="group__fields">…</div> * </section> * </div> * * -- 2. Exclusive/accordion mode (one open at a time) -- * <div class="wpbc_collapsible wpbc_collapsible--exclusive">…</div> * * -- 3. Auto-init -- * The script auto-initializes on DOMContentLoaded. No extra code needed. * * -- 4. Programmatic control (optional) * const root = document.querySelector('#wpbc_bfb__inspector'); * const api = root.__wpbc_collapsible_instance; // set by auto-init * * api.open_by_heading('Validation'); // open by heading text * api.open_by_index(0); // open the first group * * -- 5.Listen to events (e.g., to persist “open group” state) -- * root.addEventListener('wpbc:collapsible:open', (e) => { console.log( e.detail.group ); }); * root.addEventListener('wpbc:collapsible:close', (e) => { console.log( e.detail.group ); }); * * * * Markup expectations (minimal): * <div class="wpbc_collapsible [wpbc_collapsible--exclusive]"> * <section class="wpbc_ui__collapsible_group [is-open]"> * <button type="button" class="group__header"> ... </button> * <div class="group__fields"> ... </div> * </section> * ... more <section> ... * </div> * * Notes: * - Add `is-open` to any section you want initially expanded. * - Add `wpbc_collapsible--exclusive` to the container for "open one at a time" behavior. * - Works with your existing BFB markup (classes used there are the defaults). * * Accessibility: * - Sets aria-expanded on .group__header * - Sets aria-hidden + [hidden] on .group__fields * - ArrowUp/ArrowDown move focus between headers; Enter/Space toggles * * Events (bubbles from the <section>): * - 'wpbc:collapsible:open' (detail: { group, root, instance }) * - 'wpbc:collapsible:close' (detail: { group, root, instance }) * * Public API (instance methods): * - init(), destroy(), refresh() * - expand(group, [exclusive]), collapse(group), toggle(group) * - open_by_index(index), open_by_heading(text) * - is_exclusive(), is_open(group) * * @version 2025-08-26 * @since 2025-08-26 */ // --------------------------------------------------------------------------------------------------------------------- // == File /collapsible_groups.js == Time point: 2025-08-26 14:13 // --------------------------------------------------------------------------------------------------------------------- (function (w, d) { 'use strict'; class WPBC_Collapsible_Groups { /** * Create a collapsible controller for a container. * * @param {HTMLElement|string} root_el * The container element (or CSS selector) that wraps collapsible groups. * The container usually has the class `.wpbc_collapsible`. * @param {Object} [opts={}] * @param {string} [opts.group_selector='.wpbc_ui__collapsible_group'] * Selector for each collapsible group inside the container. * @param {string} [opts.header_selector='.group__header'] * Selector for the clickable header inside a group. * @param {string} [opts.fields_selector='.group__fields'] * Selector for the content/panel element inside a group. * @param {string} [opts.open_class='is-open'] * Class name that indicates the group is open. * @param {boolean} [opts.exclusive=false] * If true, only one group can be open at a time in this container. * * @constructor * @since 2025-08-26 */ constructor(root_el, opts = {}) { this.root = (typeof root_el === 'string') ? d.querySelector( root_el ) : root_el; this.opts = Object.assign( { group_selector : '.wpbc_ui__collapsible_group', header_selector: '.group__header', fields_selector: '.group__fields', open_class : 'is-open', exclusive : false }, opts ); // Bound handlers (for add/removeEventListener symmetry). /** @private */ this._on_click = this._on_click.bind( this ); /** @private */ this._on_keydown = this._on_keydown.bind( this ); /** @type {HTMLElement[]} @private */ this._groups = []; /** @type {MutationObserver|null} @private */ this._observer = null; } /** * Initialize the controller: cache groups, attach listeners, set ARIA, * and start observing DOM changes inside the container. * * @returns {WPBC_Collapsible_Groups} The instance (chainable). * @listens click * @listens keydown * @since 2025-08-26 */ init() { if ( !this.root ) { return this; } this._groups = Array.prototype.slice.call( this.root.querySelectorAll( this.opts.group_selector ) ); this.root.addEventListener( 'click', this._on_click, false ); this.root.addEventListener( 'keydown', this._on_keydown, false ); // Observe dynamic inserts/removals (Inspector re-renders). this._observer = new MutationObserver( () => { this.refresh(); } ); this._observer.observe( this.root, { childList: true, subtree: true } ); this._sync_all_aria(); return this; } /** * Tear down the controller: detach listeners, stop the observer, * and drop internal references. * * @returns {void} * @since 2025-08-26 */ destroy() { if ( !this.root ) { return; } this.root.removeEventListener( 'click', this._on_click, false ); this.root.removeEventListener( 'keydown', this._on_keydown, false ); if ( this._observer ) { this._observer.disconnect(); this._observer = null; } this._groups = []; } /** * Re-scan the DOM for current groups and re-apply ARIA to all of them. * Useful after dynamic (re)renders. * * @returns {void} * @since 2025-08-26 */ refresh() { if ( !this.root ) { return; } this._groups = Array.prototype.slice.call( this.root.querySelectorAll( this.opts.group_selector ) ); this._sync_all_aria(); } /** * Check whether the container is in exclusive (accordion) mode. * * Order of precedence: * 1) Explicit option `opts.exclusive` * 2) Container has class `.wpbc_collapsible--exclusive` * 3) Container matches `[data-wpbc-accordion="exclusive"]` * * @returns {boolean} True if exclusive mode is active. * @since 2025-08-26 */ is_exclusive() { return !!( this.opts.exclusive || this.root.classList.contains( 'wpbc_collapsible--exclusive' ) || this.root.matches( '[data-wpbc-accordion="exclusive"]' ) ); } /** * Determine whether a specific group is open. * * @param {HTMLElement} group The group element to test. * @returns {boolean} True if the group is currently open. * @since 2025-08-26 */ is_open(group) { return group.classList.contains( this.opts.open_class ); } /** * Open a group. Honors exclusive mode by collapsing all sibling groups * (queried from the live DOM at call-time). * * @param {HTMLElement} group The group element to open. * @param {boolean} [exclusive] * If provided, overrides container mode for this action only. * @returns {void} * @fires CustomEvent#wpbc:collapsible:open * @since 2025-08-26 */ expand(group, exclusive) { if ( !group ) { return; } const do_exclusive = (typeof exclusive === 'boolean') ? exclusive : this.is_exclusive(); if ( do_exclusive ) { // Always use the live DOM, not the cached list. Array.prototype.forEach.call( this.root.querySelectorAll( this.opts.group_selector ), (g) => { if ( g !== group ) { this._set_open( g, false ); } } ); } this._set_open( group, true ); } /** * Close a group. * * @param {HTMLElement} group The group element to close. * @returns {void} * @fires CustomEvent#wpbc:collapsible:close * @since 2025-08-26 */ collapse(group) { if ( !group ) { return; } this._set_open( group, false ); } /** * Toggle a group's open/closed state. * * @param {HTMLElement} group The group element to toggle. * @returns {void} * @since 2025-08-26 */ toggle(group) { if ( !group ) { return; } this[this.is_open( group ) ? 'collapse' : 'expand']( group ); } /** * Open a group by its index within the container (0-based). * * @param {number} index Zero-based index of the group. * @returns {void} * @since 2025-08-26 */ open_by_index(index) { const group = this._groups[index]; if ( group ) { this.expand( group ); } } /** * Open a group by matching text contained within the <h3> inside the header. * The comparison is case-insensitive and substring-based. * * @param {string} text Text to match against the heading contents. * @returns {void} * @since 2025-08-26 */ open_by_heading(text) { if ( !text ) { return; } const t = String( text ).toLowerCase(); const match = this._groups.find( (g) => { const h = g.querySelector( this.opts.header_selector + ' h3' ); return h && h.textContent.toLowerCase().indexOf( t ) !== -1; } ); if ( match ) { this.expand( match ); } } // ------------------------------------------------------------------------------------------------------------- // Internal // ------------------------------------------------------------------------------------------------------------- /** * Delegated click handler for headers. * * @private * @param {MouseEvent} ev The click event. * @returns {void} * @since 2025-08-26 */ _on_click(ev) { const btn = ev.target.closest( this.opts.header_selector ); if ( !btn || !this.root.contains( btn ) ) { return; } ev.preventDefault(); ev.stopPropagation(); const group = btn.closest( this.opts.group_selector ); if ( group ) { this.toggle( group ); } } /** * Keyboard handler for header interactions and roving focus: * - Enter/Space toggles the active group. * - ArrowUp/ArrowDown moves focus between group headers. * * @private * @param {KeyboardEvent} ev The keyboard event. * @returns {void} * @since 2025-08-26 */ _on_keydown(ev) { const btn = ev.target.closest( this.opts.header_selector ); if ( !btn ) { return; } const key = ev.key; // Toggle on Enter / Space. if ( key === 'Enter' || key === ' ' ) { ev.preventDefault(); const group = btn.closest( this.opts.group_selector ); if ( group ) { this.toggle( group ); } return; } // Move focus with ArrowUp/ArrowDown between headers in this container. if ( key === 'ArrowUp' || key === 'ArrowDown' ) { ev.preventDefault(); const headers = Array.prototype.map.call( this.root.querySelectorAll( this.opts.group_selector ), (g) => g.querySelector( this.opts.header_selector ) ).filter( Boolean ); const idx = headers.indexOf( btn ); if ( idx !== -1 ) { const next_idx = (key === 'ArrowDown') ? Math.min( headers.length - 1, idx + 1 ) : Math.max( 0, idx - 1 ); headers[next_idx].focus(); } } } /** * Apply ARIA synchronization to all known groups based on their open state. * * @private * @returns {void} * @since 2025-08-26 */ _sync_all_aria() { this._groups.forEach( (g) => this._sync_group_aria( g ) ); } /** * Sync ARIA attributes and visibility on a single group. * * @private * @param {HTMLElement} group The group element to sync. * @returns {void} * @since 2025-08-26 */ _sync_group_aria(group) { const is_open = this.is_open( group ); const header = group.querySelector( this.opts.header_selector ); const panel = group.querySelector( this.opts.fields_selector ); if ( header ) { // Header is a real <button>, role is harmless here. header.setAttribute( 'role', 'button' ); header.setAttribute( 'aria-expanded', is_open ? 'true' : 'false' ); // Link header to panel by id if available. if ( panel ) { if ( !panel.id ) { panel.id = this._generate_id( 'wpbc_collapsible_panel' ); } if ( !header.hasAttribute( 'aria-controls' ) ) { header.setAttribute( 'aria-controls', panel.id ); } } } if ( panel ) { panel.hidden = !is_open; panel.setAttribute( 'aria-hidden', is_open ? 'false' : 'true' ); // Optional landmark: // panel.setAttribute('role', 'region'); // panel.setAttribute('aria-labelledby', header.id || (header.id = this._generate_id('wpbc_collapsible_header'))); } } /** * Internal state change: set a group's open/closed state, sync ARIA, * manage focus on collapse, and emit a custom event. * * @private * @param {HTMLElement} group The group element to mutate. * @param {boolean} open Whether the group should be open. * @returns {void} * @fires CustomEvent#wpbc:collapsible:open * @fires CustomEvent#wpbc:collapsible:close * @since 2025-08-26 */ _set_open(group, open) { if ( !open && group.contains( document.activeElement ) ) { const header = group.querySelector( this.opts.header_selector ); header && header.focus(); } group.classList.toggle( this.opts.open_class, open ); this._sync_group_aria( group ); const ev_name = open ? 'wpbc:collapsible:open' : 'wpbc:collapsible:close'; group.dispatchEvent( new CustomEvent( ev_name, { bubbles: true, detail : { group, root: this.root, instance: this } } ) ); } /** * Generate a unique DOM id with the specified prefix. * * @private * @param {string} prefix The id prefix to use. * @returns {string} A unique element id not present in the document. * @since 2025-08-26 */ _generate_id(prefix) { let i = 1; let id; do { id = prefix + '_' + (i++); } while ( d.getElementById( id ) ); return id; } } /** * Auto-initialize collapsible controllers on the page. * Finds top-level `.wpbc_collapsible` containers (ignoring nested ones), * and instantiates {@link WPBC_Collapsible_Groups} on each. * * @function WPBC_Collapsible_AutoInit * @returns {void} * @since 2025-08-26 * @example * // Runs automatically on DOMContentLoaded; can also be called manually: * WPBC_Collapsible_AutoInit(); */ function wpbc_collapsible__auto_init() { var ROOT = '.wpbc_collapsible'; var nodes = Array.prototype.slice.call( d.querySelectorAll( ROOT ) ) .filter( function (n) { return !n.parentElement || !n.parentElement.closest( ROOT ); } ); nodes.forEach( function (node) { if ( node.__wpbc_collapsible_instance ) { return; } var exclusive = node.classList.contains( 'wpbc_collapsible--exclusive' ) || node.matches( '[data-wpbc-accordion="exclusive"]' ); node.__wpbc_collapsible_instance = new WPBC_Collapsible_Groups( node, { exclusive } ).init(); } ); } // Export to global for manual control if needed. w.WPBC_Collapsible_Groups = WPBC_Collapsible_Groups; w.WPBC_Collapsible_AutoInit = wpbc_collapsible__auto_init; // DOM-ready auto init. if ( d.readyState === 'loading' ) { d.addEventListener( 'DOMContentLoaded', wpbc_collapsible__auto_init, { once: true } ); } else { wpbc_collapsible__auto_init(); } })( window, document ); //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["ui_elements.js","ui_loading_spin.js","ui_radio_container.js","ui_full_screen_mode.js","gmail_checkbox_selection.js","bookings_checkbox_selection.js","ui_sidebar_left__actions.js","copy_text_to_clipbrd.js","collapsible_groups.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACrHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"wpbc_all_admin.js","sourcesContent":["\r\n/**\r\n * Blink specific HTML element to set attention to this element.\r\n *\r\n * @param {string} element_to_blink\t\t  - class or id of element: '.wpbc_widget_available_unavailable'\r\n * @param {int} how_many_times\t\t\t  - 4\r\n * @param {int} how_long_to_blink\t\t  - 350\r\n */\r\nfunction wpbc_blink_element( element_to_blink, how_many_times = 4, how_long_to_blink = 350 ){\r\n\r\n\tfor ( let i = 0; i < how_many_times; i++ ){\r\n\t\tjQuery( element_to_blink ).fadeOut( how_long_to_blink ).fadeIn( how_long_to_blink );\r\n\t}\r\n    jQuery( element_to_blink ).animate( {opacity: 1}, 500 );\r\n}\r\n","/**\r\n *   Support Functions - Spin Icon in Buttons  ------------------------------------------------------------------ */\r\n\r\n/**\r\n * Remove spin icon from  button and Enable this button.\r\n *\r\n * @param button_clicked_element_id\t\t- HTML ID attribute of this button\r\n * @return string\t\t\t\t\t\t- CSS classes that was previously in button icon\r\n */\r\nfunction wpbc_button__remove_spin(button_clicked_element_id) {\r\n\r\n\tvar previos_classes = '';\r\n\tif (\r\n\t\t(undefined != button_clicked_element_id)\r\n\t\t&& ('' != button_clicked_element_id)\r\n\t) {\r\n\t\tvar jElement = jQuery( '#' + button_clicked_element_id );\r\n\t\tif ( jElement.length ) {\r\n\t\t\tprevios_classes = wpbc_button_disable_loading_icon( jElement.get( 0 ) );\r\n\t\t}\r\n\t}\r\n\r\n\treturn previos_classes;\r\n}\r\n\r\n\r\n/**\r\n * Show Loading (rotating arrow) icon for button that has been clicked\r\n *\r\n * @param this_button\t\t- this object of specific button\r\n * @return string\t\t\t- CSS classes that was previously in button icon\r\n */\r\nfunction wpbc_button_enable_loading_icon(this_button) {\r\n\r\n\tvar jButton         = jQuery( this_button );\r\n\tvar jIcon           = jButton.find( 'i' );\r\n\tvar previos_classes = jIcon.attr( 'class' );\r\n\r\n\tjIcon.removeClass().addClass( 'menu_icon icon-1x wpbc_icn_rotate_right wpbc_spin' );\t// Set Rotate icon.\r\n\t// jIcon.addClass( 'wpbc_animation_pause' );\t\t\t\t\t\t\t\t\t\t\t\t// Pause animation.\r\n\t// jIcon.addClass( 'wpbc_ui_red' );\t\t\t\t\t\t\t\t\t\t\t\t\t\t// Set icon color red.\r\n\r\n\tjIcon.attr( 'wpbc_previous_class', previos_classes )\r\n\r\n\tjButton.addClass( 'disabled' );\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t// Disable button\r\n\t// We need to  set  here attr instead of prop, because for A elements,  attribute 'disabled' do  not added with jButton.prop( \"disabled\", true );.\r\n\r\n\tjButton.attr( 'wpbc_previous_onclick', jButton.attr( 'onclick' ) );\t\t// Save this value.\r\n\tjButton.attr( 'onclick', '' );\t\t\t\t\t\t\t\t\t\t\t// Disable actions \"on click\".\r\n\r\n\treturn previos_classes;\r\n}\r\n\r\n\r\n/**\r\n * Hide Loading (rotating arrow) icon for button that was clicked and show previous icon and enable button\r\n *\r\n * @param this_button\t\t- this object of specific button\r\n * @return string\t\t\t- CSS classes that was previously in button icon\r\n */\r\nfunction wpbc_button_disable_loading_icon(this_button) {\r\n\r\n\tvar jButton = jQuery( this_button );\r\n\tvar jIcon   = jButton.find( 'i' );\r\n\r\n\tvar previos_classes = jIcon.attr( 'wpbc_previous_class' );\r\n\tif (\r\n\t\t(undefined != previos_classes)\r\n\t\t&& ('' != previos_classes)\r\n\t) {\r\n\t\tjIcon.removeClass().addClass( previos_classes );\r\n\t}\r\n\r\n\tjButton.removeClass( 'disabled' );\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t// Remove Disable button.\r\n\r\n\tvar previous_onclick = jButton.attr( 'wpbc_previous_onclick' )\r\n\tif (\r\n\t\t(undefined != previous_onclick)\r\n\t\t&& ('' != previous_onclick)\r\n\t) {\r\n\t\tjButton.attr( 'onclick', previous_onclick );\r\n\t}\r\n\r\n\treturn previos_classes;\r\n}\r\n","/**\r\n * On selection  of radio button, adjust attributes of radio container\r\n *\r\n * @param _this\r\n */\r\nfunction wpbc_ui_el__radio_container_selection(_this) {\r\n\r\n\tif ( jQuery( _this ).is( ':checked' ) ) {\r\n\t\tjQuery( _this ).parents( '.wpbc_ui_radio_section' ).find( '.wpbc_ui_radio_container' ).removeAttr( 'data-selected' );\r\n\t\tjQuery( _this ).parents( '.wpbc_ui_radio_container:not(.disabled)' ).attr( 'data-selected', true );\r\n\t}\r\n\r\n\tif ( jQuery( _this ).is( ':disabled' ) ) {\r\n\t\tjQuery( _this ).parents( '.wpbc_ui_radio_container' ).addClass( 'disabled' );\r\n\t}\r\n}\r\n\r\n/**\r\n * On click on Radio Container, we will  select  the  radio button    and then adjust attributes of radio container\r\n *\r\n * @param _this\r\n */\r\nfunction wpbc_ui_el__radio_container_click(_this) {\r\n\r\n\tif ( jQuery( _this ).hasClass( 'disabled' ) ) {\r\n\t\treturn false;\r\n\t}\r\n\r\n\tvar j_radio = jQuery( _this ).find( 'input[type=radio]:not(.wpbc-form-radio-internal)' );\r\n\tif ( j_radio.length ) {\r\n\t\tj_radio.prop( 'checked', true ).trigger( 'change' );\r\n\t}\r\n\r\n}","\"use strict\";\r\n// =====================================================================================================================\r\n// == Full Screen  -  support functions   ==\r\n// =====================================================================================================================\r\n\r\n/**\r\n * Check Full  screen mode,  by  removing top tab\r\n */\r\nfunction wpbc_check_full_screen_mode(){\r\n\tif ( jQuery( 'body' ).hasClass( 'wpbc_admin_full_screen' ) ) {\r\n\t\tjQuery( 'html' ).removeClass( 'wp-toolbar' );\r\n\t} else {\r\n\t\tjQuery( 'html' ).addClass( 'wp-toolbar' );\r\n\t}\r\n\twpbc_check_buttons_max_min_in_full_screen_mode();\r\n}\r\n\r\nfunction wpbc_check_buttons_max_min_in_full_screen_mode() {\r\n\tif ( jQuery( 'body' ).hasClass( 'wpbc_admin_full_screen' ) ) {\r\n\t\tjQuery( '.wpbc_ui__top_nav__btn_full_screen'   ).addClass(    'wpbc_ui__hide' );\r\n\t\tjQuery( '.wpbc_ui__top_nav__btn_normal_screen' ).removeClass( 'wpbc_ui__hide' );\r\n\t} else {\r\n\t\tjQuery( '.wpbc_ui__top_nav__btn_full_screen'   ).removeClass( 'wpbc_ui__hide' );\r\n\t\tjQuery( '.wpbc_ui__top_nav__btn_normal_screen' ).addClass(    'wpbc_ui__hide' );\r\n\t}\r\n}\r\n\r\njQuery( document ).ready( function () {\r\n\twpbc_check_full_screen_mode();\r\n} );","/**\r\n * Checkbox Selection functions for Listing.\r\n */\r\n\r\n/**\r\n * Selections of several  checkboxes like in gMail with shift :)\r\n * Need to  have this structure:\r\n * .wpbc_selectable_table\r\n *      .wpbc_selectable_head\r\n *              .check-column\r\n *                  :checkbox\r\n *      .wpbc_selectable_body\r\n *          .wpbc_row\r\n *              .check-column\r\n *                  :checkbox\r\n *      .wpbc_selectable_foot\r\n *              .check-column\r\n *                  :checkbox\r\n */\r\nfunction wpbc_define_gmail_checkbox_selection( $ ){\r\n\r\n\tvar checks, first, last, checked, sliced, lastClicked = false;\r\n\r\n\t// Check all checkboxes.\r\n\t$( '.wpbc_selectable_body' ).find( '.check-column' ).find( ':checkbox' ).on(\r\n\t\t'click',\r\n\t\tfunction (e) {\r\n\t\t\tif ( 'undefined' == e.shiftKey ) {\r\n\t\t\t\treturn true;\r\n\t\t\t}\r\n\t\t\tif ( e.shiftKey ) {\r\n\t\t\t\tif ( ! lastClicked ) {\r\n\t\t\t\t\treturn true;\r\n\t\t\t\t}\r\n\t\t\t\tchecks  = $( lastClicked ).closest( '.wpbc_selectable_body' ).find( ':checkbox' ).filter( ':visible:enabled' );\r\n\t\t\t\tfirst   = checks.index( lastClicked );\r\n\t\t\t\tlast    = checks.index( this );\r\n\t\t\t\tchecked = $( this ).prop( 'checked' );\r\n\t\t\t\tif ( 0 < first && 0 < last && first != last ) {\r\n\t\t\t\t\tsliced = (last > first) ? checks.slice( first, last ) : checks.slice( last, first );\r\n\t\t\t\t\tsliced.prop(\r\n\t\t\t\t\t\t'checked',\r\n\t\t\t\t\t\tfunction () {\r\n\t\t\t\t\t\t\tif ( $( this ).closest( '.wpbc_row' ).is( ':visible' ) ) {\r\n\t\t\t\t\t\t\t\treturn checked;\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\treturn false;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t).trigger( 'change' );\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t\tlastClicked = this;\r\n\r\n\t\t\t// toggle \"check all\" checkboxes.\r\n\t\t\tvar unchecked = $( this ).closest( '.wpbc_selectable_body' ).find( ':checkbox' ).filter( ':visible:enabled' ).not( ':checked' );\r\n\t\t\t$( this ).closest( '.wpbc_selectable_table' ).children( '.wpbc_selectable_head, .wpbc_selectable_foot' ).find( ':checkbox' ).prop(\r\n\t\t\t\t'checked',\r\n\t\t\t\tfunction () {\r\n\t\t\t\t\treturn (0 === unchecked.length);\r\n\t\t\t\t}\r\n\t\t\t).trigger( 'change' );\r\n\r\n\t\t\treturn true;\r\n\t\t}\r\n\t);\r\n\r\n\t// Head || Foot clicking to  select / deselect ALL.\r\n\t$( '.wpbc_selectable_head, .wpbc_selectable_foot' ).find( '.check-column :checkbox' ).on(\r\n\t\t'click',\r\n\t\tfunction (event) {\r\n\t\t\tvar $this          = $( this ),\r\n\t\t\t\t$table         = $this.closest( '.wpbc_selectable_table' ),\r\n\t\t\t\tcontrolChecked = $this.prop( 'checked' ),\r\n\t\t\t\ttoggle         = event.shiftKey || $this.data( 'wp-toggle' );\r\n\r\n\t\t\t$table.children( '.wpbc_selectable_body' ).filter( ':visible' )\r\n\t\t\t\t.find( '.check-column' ).find( ':checkbox' )\r\n\t\t\t\t.prop(\r\n\t\t\t\t\t'checked',\r\n\t\t\t\t\tfunction () {\r\n\t\t\t\t\t\tif ( $( this ).is( ':hidden,:disabled' ) ) {\r\n\t\t\t\t\t\t\treturn false;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\tif ( toggle ) {\r\n\t\t\t\t\t\t\treturn ! $( this ).prop( 'checked' );\r\n\t\t\t\t\t\t} else if ( controlChecked ) {\r\n\t\t\t\t\t\t\treturn true;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\treturn false;\r\n\t\t\t\t\t}\r\n\t\t\t\t).trigger( 'change' );\r\n\r\n\t\t\t$table.children( '.wpbc_selectable_head,  .wpbc_selectable_foot' ).filter( ':visible' )\r\n\t\t\t\t.find( '.check-column' ).find( ':checkbox' )\r\n\t\t\t\t.prop(\r\n\t\t\t\t\t'checked',\r\n\t\t\t\t\tfunction () {\r\n\t\t\t\t\t\tif ( toggle ) {\r\n\t\t\t\t\t\t\treturn false;\r\n\t\t\t\t\t\t} else if ( controlChecked ) {\r\n\t\t\t\t\t\t\treturn true;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\treturn false;\r\n\t\t\t\t\t}\r\n\t\t\t\t);\r\n\t\t}\r\n\t);\r\n\r\n\r\n\t// Visually  show selected border.\r\n\t$( '.wpbc_selectable_body' ).find( '.check-column :checkbox' ).on(\r\n\t\t'change',\r\n\t\tfunction (event) {\r\n\t\t\tif ( jQuery( this ).is( ':checked' ) ) {\r\n\t\t\t\tjQuery( this ).closest( '.wpbc_list_row' ).addClass( 'row_selected_color' );\r\n\t\t\t} else {\r\n\t\t\t\tjQuery( this ).closest( '.wpbc_list_row' ).removeClass( 'row_selected_color' );\r\n\t\t\t}\r\n\r\n\t\t\t// Disable text selection while pressing 'shift'.\r\n\t\t\tdocument.getSelection().removeAllRanges();\r\n\r\n\t\t\t// Show or hide buttons on Actions toolbar  at  Booking Listing  page,  if we have some selected bookings.\r\n\t\t\twpbc_show_hide_action_buttons_for_selected_bookings();\r\n\t\t}\r\n\t);\r\n\r\n\twpbc_show_hide_action_buttons_for_selected_bookings();\r\n}\r\n","\r\n/**\r\n * Get ID array  of selected elements\r\n */\r\nfunction wpbc_get_selected_row_id() {\r\n\r\n\tvar $table      = jQuery( '.wpbc__wrap__booking_listing .wpbc_selectable_table' );\r\n\tvar checkboxes  = $table.children( '.wpbc_selectable_body' ).filter( ':visible' ).find( '.check-column' ).find( ':checkbox' );\r\n\tvar selected_id = [];\r\n\r\n\tjQuery.each(\r\n\t\tcheckboxes,\r\n\t\tfunction (key, checkbox) {\r\n\t\t\tif ( jQuery( checkbox ).is( ':checked' ) ) {\r\n\t\t\t\tvar element_id = wpbc_get_row_id_from_element( checkbox );\r\n\t\t\t\tselected_id.push( element_id );\r\n\t\t\t}\r\n\t\t}\r\n\t);\r\n\r\n\treturn selected_id;\r\n}\r\n\r\n\r\n/**\r\n * Get ID of row,  based on clciked element\r\n *\r\n * @param this_inbound_element  - ususlly  this\r\n * @returns {number}\r\n */\r\nfunction wpbc_get_row_id_from_element(this_inbound_element) {\r\n\r\n\tvar element_id = jQuery( this_inbound_element ).closest( '.wpbc_listing_usual_row' ).attr( 'id' );\r\n\r\n\telement_id = parseInt( element_id.replace( 'row_id_', '' ) );\r\n\r\n\treturn element_id;\r\n}\r\n\r\n\r\n/**\r\n * == Booking Listing == Show or hide buttons on Actions toolbar  at    page,  if we have some selected bookings.\r\n */\r\nfunction wpbc_show_hide_action_buttons_for_selected_bookings(){\r\n\r\n\tvar selected_rows_arr = wpbc_get_selected_row_id();\r\n\r\n\tif ( selected_rows_arr.length > 0 ) {\r\n\t\tjQuery( '.hide_button_if_no_selection' ).show();\r\n\t} else {\r\n\t\tjQuery( '.hide_button_if_no_selection' ).hide();\r\n\t}\r\n}","\"use strict\";\r\n// =====================================================================================================================\r\n// == Left Bar  -  expand / colapse functions   ==\r\n// =====================================================================================================================\r\n\r\n/**\r\n * Expand Vertical Left Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_left__do_max() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'max' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\r\n\tjQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' );\r\n\tjQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_max' );\r\n}\r\n\r\n/**\r\n * Hide Vertical Left Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_left__do_min() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'min' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n\r\n\tjQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' );\r\n\tjQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_min' );\r\n}\r\n\r\n/**\r\n * Colapse Vertical Left Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_left__do_compact() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'compact' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n\r\n\tjQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' );\r\n\tjQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_compact' );\r\n}\r\n\r\n/**\r\n * Completely Hide Vertical Left Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_left__do_hide() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min max compact none' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'none' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_left_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_left_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n\t// Hide top \"Menu\" button with divider.\r\n\tjQuery( '.wpbc_ui__top_nav__btn_show_left_vertical_nav,.wpbc_ui__top_nav__btn_show_left_vertical_nav_divider' ).addClass( 'wpbc_ui__hide' );\r\n\r\n\tjQuery( '.wp-admin' ).removeClass( 'wpbc_page_wrapper_left_min wpbc_page_wrapper_left_max wpbc_page_wrapper_left_compact wpbc_page_wrapper_left_none' );\r\n\tjQuery( '.wp-admin' ).addClass( 'wpbc_page_wrapper_left_none' );\r\n}\r\n\r\n/**\r\n * Action on click \"Go Back\" - show root menu\r\n * or some other section in left sidebar.\r\n *\r\n * @param string menu_to_show - menu slug.\r\n */\r\nfunction wpbc_admin_ui__sidebar_left__show_section( menu_to_show ) {\r\n\tjQuery( '.wpbc_ui_el__vert_left_bar__section' ).addClass( 'wpbc_ui__hide' )\r\n\tjQuery( '.wpbc_ui_el__vert_left_bar__section_' + menu_to_show ).removeClass( 'wpbc_ui__hide' );\r\n}\r\n\r\n// =====================================================================================================================\r\n// == Right Side Bar  -  expand / colapse functions   ==\r\n// =====================================================================================================================\r\n\r\n/**\r\n * Expand Vertical Right Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_right__do_max() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'max_right' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n}\r\n\r\n/**\r\n * Hide Vertical Right Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_right__do_min() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'min_right' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n}\r\n\r\n/**\r\n * Colapse Vertical Right Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_right__do_compact() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'compact_right' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n}\r\n\r\n/**\r\n * Completely Hide Vertical Right Bar.\r\n */\r\nfunction wpbc_admin_ui__sidebar_right__do_hide() {\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).removeClass( 'min_right max_right compact_right none_right' );\r\n\tjQuery( '.wpbc_settings_page_wrapper' ).addClass( 'none_right' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_open_right_vertical_nav' ).removeClass( 'wpbc_ui__hide' );\r\n\tjQuery( '.wpbc_ui__top_nav__btn_hide_right_vertical_nav' ).addClass( 'wpbc_ui__hide' );\r\n\t// Hide top \"Menu\" button with divider.\r\n\tjQuery( '.wpbc_ui__top_nav__btn_show_right_vertical_nav,.wpbc_ui__top_nav__btn_show_right_vertical_nav_divider' ).addClass( 'wpbc_ui__hide' );\r\n}\r\n\r\n/**\r\n * Action on click \"Go Back\" - show root menu\r\n * or some other section in right sidebar.\r\n *\r\n * @param string menu_to_show - menu slug.\r\n */\r\nfunction wpbc_admin_ui__sidebar_right__show_section( menu_to_show ) {\r\n\tjQuery( '.wpbc_ui_el__vert_right_bar__section' ).addClass( 'wpbc_ui__hide' )\r\n\tjQuery( '.wpbc_ui_el__vert_right_bar__section_' + menu_to_show ).removeClass( 'wpbc_ui__hide' );\r\n}\r\n\r\n// =====================================================================================================================\r\n// == End Right Side Bar  section   ==\r\n// =====================================================================================================================\r\n\r\n/**\r\n * Get anchor(s) array  from  URL.\r\n * Doc: https://developer.mozilla.org/en-US/docs/Web/API/Location\r\n *\r\n * @returns {*[]}\r\n */\r\nfunction wpbc_url_get_anchors_arr() {\r\n\tvar hashes            = window.location.hash.replace( '%23', '#' );\r\n\tvar hashes_arr        = hashes.split( '#' );\r\n\tvar result            = [];\r\n\tvar hashes_arr_length = hashes_arr.length;\r\n\r\n\tfor ( var i = 0; i < hashes_arr_length; i++ ) {\r\n\t\tif ( hashes_arr[i].length > 0 ) {\r\n\t\t\tresult.push( hashes_arr[i] );\r\n\t\t}\r\n\t}\r\n\treturn result;\r\n}\r\n\r\n/**\r\n * Auto Expand Settings section based on URL anchor, after  page loaded.\r\n */\r\njQuery( document ).ready( function () { wpbc_admin_ui__do_expand_section(); setTimeout( 'wpbc_admin_ui__do_expand_section', 10 ); } );\r\njQuery( document ).ready( function () { wpbc_admin_ui__do_expand_section(); setTimeout( 'wpbc_admin_ui__do_expand_section', 150 ); } );\r\n\r\n/**\r\n * Expand section in  General Settings page and select Menu item.\r\n */\r\nfunction wpbc_admin_ui__do_expand_section() {\r\n\r\n\t// window.location.hash  = #section_id  /  doc: https://developer.mozilla.org/en-US/docs/Web/API/Location .\r\n\tvar anchors_arr        = wpbc_url_get_anchors_arr();\r\n\tvar anchors_arr_length = anchors_arr.length;\r\n\r\n\tif ( anchors_arr_length > 0 ) {\r\n\t\tvar one_anchor_prop_value = anchors_arr[0].split( 'do_expand__' );\r\n\t\tif ( one_anchor_prop_value.length > 1 ) {\r\n\r\n\t\t\t// 'wpbc_general_settings_calendar_metabox'\r\n\t\t\tvar section_to_show    = one_anchor_prop_value[1];\r\n\t\t\tvar section_id_to_show = '#' + section_to_show;\r\n\r\n\r\n\t\t\t// -- Remove selected background in all left  menu  items ---------------------------------------------------\r\n\t\t\tjQuery( '.wpbc_ui_el__vert_nav_item ' ).removeClass( 'active' );\r\n\t\t\t// Set left menu selected.\r\n\t\t\tjQuery( '.do_expand__' + section_to_show + '_link' ).addClass( 'active' );\r\n\t\t\tvar selected_title = jQuery( '.do_expand__' + section_to_show + '_link a .wpbc_ui_el__vert_nav_title ' ).text();\r\n\r\n\t\t\t// Expand section, if it colapsed.\r\n\t\t\tif ( ! jQuery( '.do_expand__' + section_to_show + '_link' ).parents( '.wpbc_ui_el__level__folder' ).hasClass( 'expanded' ) ) {\r\n\t\t\t\tjQuery( '.wpbc_ui_el__level__folder' ).removeClass( 'expanded' );\r\n\t\t\t\tjQuery( '.do_expand__' + section_to_show + '_link' ).parents( '.wpbc_ui_el__level__folder' ).addClass( 'expanded' );\r\n\t\t\t}\r\n\r\n\t\t\t// -- Expand section ---------------------------------------------------------------------------------------\r\n\t\t\tvar container_to_hide_class = '.postbox';\r\n\t\t\t// Hide sections '.postbox' in admin page and show specific one.\r\n\t\t\tjQuery( '.wpbc_admin_page ' + container_to_hide_class ).hide();\r\n\t\t\tjQuery( '.wpbc_container_always_hide__on_left_nav_click' ).hide();\r\n\t\t\tjQuery( section_id_to_show ).show();\r\n\r\n\t\t\t// Show all other sections,  if provided in URL: ..?page=wpbc-settings#do_expand__wpbc_general_settings_capacity_metabox#wpbc_general_settings_capacity_upgrade_metabox .\r\n\t\t\tfor ( let i = 1; i < anchors_arr_length; i++ ) {\r\n\t\t\t\tjQuery( '#' + anchors_arr[i] ).show();\r\n\t\t\t}\r\n\r\n\t\t\tif ( false ) {\r\n\t\t\t\tvar targetOffset = wpbc_scroll_to( section_id_to_show );\r\n\t\t\t}\r\n\r\n\t\t\t// -- Set Value to Input about selected Nav element  ---------------------------------------------------------------       // FixIn: 9.8.6.1.\r\n\t\t\tvar section_id_tab = section_id_to_show.substring( 0, section_id_to_show.length - 8 ) + '_tab';\r\n\t\t\tif ( container_to_hide_class == section_id_to_show ) {\r\n\t\t\t\tsection_id_tab = '#wpbc_general_settings_all_tab'\r\n\t\t\t}\r\n\t\t\tif ( '#wpbc_general_settings_capacity_metabox,#wpbc_general_settings_capacity_upgrade_metabox' == section_id_to_show ) {\r\n\t\t\t\tsection_id_tab = '#wpbc_general_settings_capacity_tab'\r\n\t\t\t}\r\n\t\t\tjQuery( '#form_visible_section' ).val( section_id_tab );\r\n\t\t}\r\n\r\n\t\t// Like blinking some elements.\r\n\t\twpbc_admin_ui__do__anchor__another_actions();\r\n\t}\r\n}\r\n\r\nfunction wpbc_admin_ui__is_in_mobile_screen_size() {\r\n\treturn wpbc_admin_ui__is_in_this_screen_size( 605 );\r\n}\r\n\r\nfunction wpbc_admin_ui__is_in_this_screen_size(size) {\r\n\treturn (window.screen.width <= size);\r\n}\r\n\r\n/**\r\n * Open settings page  |  Expand section  |  Select Menu item.\r\n */\r\nfunction wpbc_admin_ui__do__open_url__expand_section(url, section_id) {\r\n\r\n\t// window.location.href = url + '&do_expand=' + section_id + '#do_expand__' + section_id; //.\r\n\twindow.location.href = url + '#do_expand__' + section_id;\r\n\r\n\tif ( wpbc_admin_ui__is_in_mobile_screen_size() ) {\r\n\t\twpbc_admin_ui__sidebar_left__do_min();\r\n\t}\r\n\r\n\twpbc_admin_ui__do_expand_section();\r\n}\r\n\r\n\r\n/**\r\n * Check  for Other actions:  Like blinking some elements in settings page. E.g. Days selection  or  change-over days.\r\n */\r\nfunction wpbc_admin_ui__do__anchor__another_actions() {\r\n\r\n\tvar anchors_arr        = wpbc_url_get_anchors_arr();\r\n\tvar anchors_arr_length = anchors_arr.length;\r\n\r\n\t// Other actions:  Like blinking some elements.\r\n\tfor ( var i = 0; i < anchors_arr_length; i++ ) {\r\n\r\n\t\tvar this_anchor = anchors_arr[i];\r\n\r\n\t\tvar this_anchor_prop_value = this_anchor.split( 'do_other_actions__' );\r\n\r\n\t\tif ( this_anchor_prop_value.length > 1 ) {\r\n\r\n\t\t\tvar section_action = this_anchor_prop_value[1];\r\n\r\n\t\t\tswitch ( section_action ) {\r\n\r\n\t\t\t\tcase 'blink_day_selections':\r\n\t\t\t\t\t// wpbc_ui_settings__panel__click( '#wpbc_general_settings_calendar_tab a', '#wpbc_general_settings_calendar_metabox', 'Days Selection' );.\r\n\t\t\t\t\twpbc_blink_element( '.wpbc_tr_set_gen_booking_type_of_day_selections', 4, 350 );\r\n\t\t\t\t\t\twpbc_scroll_to( '.wpbc_tr_set_gen_booking_type_of_day_selections' );\r\n\t\t\t\t\tbreak;\r\n\r\n\t\t\t\tcase 'blink_change_over_days':\r\n\t\t\t\t\t// wpbc_ui_settings__panel__click( '#wpbc_general_settings_calendar_tab a', '#wpbc_general_settings_calendar_metabox', 'Changeover Days' );.\r\n\t\t\t\t\twpbc_blink_element( '.wpbc_tr_set_gen_booking_range_selection_time_is_active', 4, 350 );\r\n\t\t\t\t\t\twpbc_scroll_to( '.wpbc_tr_set_gen_booking_range_selection_time_is_active' );\r\n\t\t\t\t\tbreak;\r\n\r\n\t\t\t\tcase 'blink_captcha':\r\n\t\t\t\t\twpbc_blink_element( '.wpbc_tr_set_gen_booking_is_use_captcha', 4, 350 );\r\n\t\t\t\t\t\twpbc_scroll_to( '.wpbc_tr_set_gen_booking_is_use_captcha' );\r\n\t\t\t\t\tbreak;\r\n\r\n\t\t\t\tdefault:\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n}","/**\r\n * Copy txt to clipbrd from Text fields.\r\n *\r\n * @param html_element_id  - e.g. 'data_field'\r\n * @returns {boolean}\r\n */\r\nfunction wpbc_copy_text_to_clipbrd_from_element( html_element_id ) {\r\n\t// Get the text field.\r\n\tvar copyText = document.getElementById( html_element_id );\r\n\r\n\t// Select the text field.\r\n\tcopyText.select();\r\n\tcopyText.setSelectionRange( 0, 99999 ); // For mobile devices.\r\n\r\n\t// Copy the text inside the text field.\r\n\tvar is_copied = wpbc_copy_text_to_clipbrd( copyText.value );\r\n\tif ( ! is_copied ) {\r\n\t\tconsole.error( 'Oops, unable to copy', copyText.value );\r\n\t}\r\n\treturn is_copied;\r\n}\r\n\r\n/**\r\n * Copy txt to clipbrd.\r\n *\r\n * @param text\r\n * @returns {boolean}\r\n */\r\nfunction wpbc_copy_text_to_clipbrd(text) {\r\n\r\n\tif ( ! navigator.clipboard ) {\r\n\t\treturn wpbc_fallback_copy_text_to_clipbrd( text );\r\n\t}\r\n\r\n\tnavigator.clipboard.writeText( text ).then(\r\n\t\tfunction () {\r\n\t\t\t// console.log( 'Async: Copying to clipboard was successful!' );.\r\n\t\t\treturn  true;\r\n\t\t},\r\n\t\tfunction (err) {\r\n\t\t\t// console.error( 'Async: Could not copy text: ', err );.\r\n\t\t\treturn  false;\r\n\t\t}\r\n\t);\r\n}\r\n\r\n/**\r\n * Copy txt to clipbrd - depricated method.\r\n *\r\n * @param text\r\n * @returns {boolean}\r\n */\r\nfunction wpbc_fallback_copy_text_to_clipbrd( text ) {\r\n\r\n\t// -----------------------------------------------------------------------------------------------------------------\r\n\t// var textArea   = document.createElement( \"textarea\" );\r\n\t// textArea.value = text;\r\n\t//\r\n\t// // Avoid scrolling to bottom.\r\n\t// textArea.style.top      = \"0\";\r\n\t// textArea.style.left     = \"0\";\r\n\t// textArea.style.position = \"fixed\";\r\n\t// textArea.style.zIndex   = \"999999999\";\r\n\t// document.body.appendChild( textArea );\r\n\t// textArea.focus();\r\n\t// textArea.select();\r\n\r\n\t// -----------------------------------------------------------------------------------------------------------------\r\n\t// Now get it as HTML  (original here https://stackoverflow.com/questions/34191780/javascript-copy-string-to-clipboard-as-text-html ).\r\n\r\n\t// [1] - Create container for the HTML.\r\n\tvar container       = document.createElement( 'div' );\r\n\tcontainer.innerHTML = text;\r\n\r\n\t// [2] - Hide element.\r\n\tcontainer.style.position      = 'fixed';\r\n\tcontainer.style.pointerEvents = 'none';\r\n\tcontainer.style.opacity       = 0;\r\n\r\n\t// Detect all style sheets of the page.\r\n\tvar activeSheets = Array.prototype.slice.call( document.styleSheets ).filter(\r\n\t\tfunction (sheet) {\r\n\t\t\treturn ! sheet.disabled;\r\n\t\t}\r\n\t);\r\n\r\n\t// [3] - Mount the container to the DOM to make `contentWindow` available.\r\n\tdocument.body.appendChild( container );\r\n\r\n\t// [4] - Copy to clipboard.\r\n\twindow.getSelection().removeAllRanges();\r\n\r\n\tvar range = document.createRange();\r\n\trange.selectNode( container );\r\n\twindow.getSelection().addRange( range );\r\n\t// -----------------------------------------------------------------------------------------------------------------\r\n\r\n\tvar result = false;\r\n\r\n\ttry {\r\n\t\tresult = document.execCommand( 'copy' );\r\n\t\t// console.log( 'Fallback: Copying text command was ' + msg ); //.\r\n\t} catch ( err ) {\r\n\t\t// console.error( 'Fallback: Oops, unable to copy', err ); //.\r\n\t}\r\n\t// document.body.removeChild( textArea ); //.\r\n\r\n\t// [5.4] - Enable CSS.\r\n\tvar activeSheets_length = activeSheets.length;\r\n\tfor ( var i = 0; i < activeSheets_length; i++ ) {\r\n\t\tactiveSheets[i].disabled = false;\r\n\t}\r\n\r\n\t// [6] - Remove the container\r\n\tdocument.body.removeChild( container );\r\n\r\n\treturn  result;\r\n}","/**\r\n * WPBC Collapsible Groups\r\n *\r\n * Universal, dependency-free controller for expanding/collapsing grouped sections in right-side panels (Inspector/Library/Form Settings, or any other WPBC page).\r\n *\r\n * \t\t=== How to use it (quick) ? ===\r\n *\r\n *\t\t-- 1. Markup (independent mode: multiple open allowed) --\r\n *\t\t\t<div class=\"wpbc_collapsible\">\r\n *\t\t\t  <section class=\"wpbc_ui__collapsible_group is-open\">\r\n *\t\t\t\t<button type=\"button\" class=\"group__header\"><h3>General</h3></button>\r\n *\t\t\t\t<div class=\"group__fields\">…</div>\r\n *\t\t\t  </section>\r\n *\t\t\t  <section class=\"wpbc_ui__collapsible_group\">\r\n *\t\t\t\t<button type=\"button\" class=\"group__header\"><h3>Advanced</h3></button>\r\n *\t\t\t\t<div class=\"group__fields\">…</div>\r\n *\t\t\t  </section>\r\n *\t\t\t</div>\r\n *\r\n *\t\t-- 2. Exclusive/accordion mode (one open at a time) --\r\n *\t\t\t<div class=\"wpbc_collapsible wpbc_collapsible--exclusive\">…</div>\r\n *\r\n *\t\t-- 3. Auto-init --\r\n *\t\t\tThe script auto-initializes on DOMContentLoaded. No extra code needed.\r\n *\r\n *\t\t-- 4. Programmatic control (optional)\r\n *\t\t\tconst root = document.querySelector('#wpbc_bfb__inspector');\r\n *\t\t\tconst api  = root.__wpbc_collapsible_instance; // set by auto-init\r\n *\r\n *\t\t\tapi.open_by_heading('Validation'); // open by heading text\r\n *\t\t\tapi.open_by_index(0);              // open the first group\r\n *\r\n *\t\t-- 5.Listen to events (e.g., to persist “open group” state) --\r\n *\t\t\troot.addEventListener('wpbc:collapsible:open',  (e) => { console.log(  e.detail.group ); });\r\n *\t\t\troot.addEventListener('wpbc:collapsible:close', (e) => { console.log(  e.detail.group ); });\r\n *\r\n *\r\n *\r\n * Markup expectations (minimal):\r\n *  <div class=\"wpbc_collapsible [wpbc_collapsible--exclusive]\">\r\n *    <section class=\"wpbc_ui__collapsible_group [is-open]\">\r\n *      <button type=\"button\" class=\"group__header\"> ... </button>\r\n *      <div class=\"group__fields\"> ... </div>\r\n *    </section>\r\n *    ... more <section> ...\r\n *  </div>\r\n *\r\n * Notes:\r\n *  - Add `is-open` to any section you want initially expanded.\r\n *  - Add `wpbc_collapsible--exclusive` to the container for \"open one at a time\" behavior.\r\n *  - Works with your existing BFB markup (classes used there are the defaults).\r\n *\r\n * Accessibility:\r\n *  - Sets aria-expanded on .group__header\r\n *  - Sets aria-hidden + [hidden] on .group__fields\r\n *  - ArrowUp/ArrowDown move focus between headers; Enter/Space toggles\r\n *\r\n * Events (bubbles from the <section>):\r\n *  - 'wpbc:collapsible:open'  (detail: { group, root, instance })\r\n *  - 'wpbc:collapsible:close' (detail: { group, root, instance })\r\n *\r\n * Public API (instance methods):\r\n *  - init(), destroy(), refresh()\r\n *  - expand(group, [exclusive]), collapse(group), toggle(group)\r\n *  - open_by_index(index), open_by_heading(text)\r\n *  - is_exclusive(), is_open(group)\r\n *\r\n * @version 2025-08-26\r\n * @since 2025-08-26\r\n */\r\n// ---------------------------------------------------------------------------------------------------------------------\r\n// == File  /collapsible_groups.js == Time point: 2025-08-26 14:13\r\n// ---------------------------------------------------------------------------------------------------------------------\r\n(function (w, d) {\r\n\t'use strict';\r\n\r\n\tclass WPBC_Collapsible_Groups {\r\n\r\n\t\t/**\r\n\t\t * Create a collapsible controller for a container.\r\n\t\t *\r\n\t\t * @param {HTMLElement|string} root_el\r\n\t\t *        The container element (or CSS selector) that wraps collapsible groups.\r\n\t\t *        The container usually has the class `.wpbc_collapsible`.\r\n\t\t * @param {Object} [opts={}]\r\n\t\t * @param {string}  [opts.group_selector='.wpbc_ui__collapsible_group']\r\n\t\t *        Selector for each collapsible group inside the container.\r\n\t\t * @param {string}  [opts.header_selector='.group__header']\r\n\t\t *        Selector for the clickable header inside a group.\r\n\t\t * @param {string}  [opts.fields_selector='.group__fields']\r\n\t\t *        Selector for the content/panel element inside a group.\r\n\t\t * @param {string}  [opts.open_class='is-open']\r\n\t\t *        Class name that indicates the group is open.\r\n\t\t * @param {boolean} [opts.exclusive=false]\r\n\t\t *        If true, only one group can be open at a time in this container.\r\n\t\t *\r\n\t\t * @constructor\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\tconstructor(root_el, opts = {}) {\r\n\t\t\tthis.root = (typeof root_el === 'string') ? d.querySelector( root_el ) : root_el;\r\n\t\t\tthis.opts = Object.assign( {\r\n\t\t\t\tgroup_selector : '.wpbc_ui__collapsible_group',\r\n\t\t\t\theader_selector: '.group__header',\r\n\t\t\t\tfields_selector: '.group__fields',\r\n\t\t\t\topen_class     : 'is-open',\r\n\t\t\t\texclusive      : false\r\n\t\t\t}, opts );\r\n\r\n\t\t\t// Bound handlers (for add/removeEventListener symmetry).\r\n\t\t\t/** @private */\r\n\t\t\tthis._on_click = this._on_click.bind( this );\r\n\t\t\t/** @private */\r\n\t\t\tthis._on_keydown = this._on_keydown.bind( this );\r\n\r\n\t\t\t/** @type {HTMLElement[]} @private */\r\n\t\t\tthis._groups = [];\r\n\t\t\t/** @type {MutationObserver|null} @private */\r\n\t\t\tthis._observer = null;\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Initialize the controller: cache groups, attach listeners, set ARIA,\r\n\t\t * and start observing DOM changes inside the container.\r\n\t\t *\r\n\t\t * @returns {WPBC_Collapsible_Groups} The instance (chainable).\r\n\t\t * @listens click\r\n\t\t * @listens keydown\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\tinit() {\r\n\t\t\tif ( !this.root ) {\r\n\t\t\t\treturn this;\r\n\t\t\t}\r\n\t\t\tthis._groups = Array.prototype.slice.call(\r\n\t\t\t\tthis.root.querySelectorAll( this.opts.group_selector )\r\n\t\t\t);\r\n\t\t\tthis.root.addEventListener( 'click', this._on_click, false );\r\n\t\t\tthis.root.addEventListener( 'keydown', this._on_keydown, false );\r\n\r\n\t\t\t// Observe dynamic inserts/removals (Inspector re-renders).\r\n\t\t\tthis._observer = new MutationObserver( () => {\r\n\t\t\t\tthis.refresh();\r\n\t\t\t} );\r\n\t\t\tthis._observer.observe( this.root, { childList: true, subtree: true } );\r\n\r\n\t\t\tthis._sync_all_aria();\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Tear down the controller: detach listeners, stop the observer,\r\n\t\t * and drop internal references.\r\n\t\t *\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\tdestroy() {\r\n\t\t\tif ( !this.root ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tthis.root.removeEventListener( 'click', this._on_click, false );\r\n\t\t\tthis.root.removeEventListener( 'keydown', this._on_keydown, false );\r\n\t\t\tif ( this._observer ) {\r\n\t\t\t\tthis._observer.disconnect();\r\n\t\t\t\tthis._observer = null;\r\n\t\t\t}\r\n\t\t\tthis._groups = [];\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Re-scan the DOM for current groups and re-apply ARIA to all of them.\r\n\t\t * Useful after dynamic (re)renders.\r\n\t\t *\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\trefresh() {\r\n\t\t\tif ( !this.root ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tthis._groups = Array.prototype.slice.call(\r\n\t\t\t\tthis.root.querySelectorAll( this.opts.group_selector )\r\n\t\t\t);\r\n\t\t\tthis._sync_all_aria();\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Check whether the container is in exclusive (accordion) mode.\r\n\t\t *\r\n\t\t * Order of precedence:\r\n\t\t *  1) Explicit option `opts.exclusive`\r\n\t\t *  2) Container has class `.wpbc_collapsible--exclusive`\r\n\t\t *  3) Container matches `[data-wpbc-accordion=\"exclusive\"]`\r\n\t\t *\r\n\t\t * @returns {boolean} True if exclusive mode is active.\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\tis_exclusive() {\r\n\t\t\treturn !!(\r\n\t\t\t\tthis.opts.exclusive ||\r\n\t\t\t\tthis.root.classList.contains( 'wpbc_collapsible--exclusive' ) ||\r\n\t\t\t\tthis.root.matches( '[data-wpbc-accordion=\"exclusive\"]' )\r\n\t\t\t);\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Determine whether a specific group is open.\r\n\t\t *\r\n\t\t * @param {HTMLElement} group The group element to test.\r\n\t\t * @returns {boolean} True if the group is currently open.\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\tis_open(group) {\r\n\t\t\treturn group.classList.contains( this.opts.open_class );\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Open a group. Honors exclusive mode by collapsing all sibling groups\r\n\t\t * (queried from the live DOM at call-time).\r\n\t\t *\r\n\t\t * @param {HTMLElement} group The group element to open.\r\n\t\t * @param {boolean} [exclusive]\r\n\t\t *        If provided, overrides container mode for this action only.\r\n\t\t * @returns {void}\r\n\t\t * @fires CustomEvent#wpbc:collapsible:open\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\texpand(group, exclusive) {\r\n\t\t\tif ( !group ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tconst do_exclusive = (typeof exclusive === 'boolean') ? exclusive : this.is_exclusive();\r\n\t\t\tif ( do_exclusive ) {\r\n\t\t\t\t// Always use the live DOM, not the cached list.\r\n\t\t\t\tArray.prototype.forEach.call(\r\n\t\t\t\t\tthis.root.querySelectorAll( this.opts.group_selector ),\r\n\t\t\t\t\t(g) => {\r\n\t\t\t\t\t\tif ( g !== group ) {\r\n\t\t\t\t\t\t\tthis._set_open( g, false );\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t);\r\n\t\t\t}\r\n\t\t\tthis._set_open( group, true );\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Close a group.\r\n\t\t *\r\n\t\t * @param {HTMLElement} group The group element to close.\r\n\t\t * @returns {void}\r\n\t\t * @fires CustomEvent#wpbc:collapsible:close\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\tcollapse(group) {\r\n\t\t\tif ( !group ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tthis._set_open( group, false );\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Toggle a group's open/closed state.\r\n\t\t *\r\n\t\t * @param {HTMLElement} group The group element to toggle.\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\ttoggle(group) {\r\n\t\t\tif ( !group ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tthis[this.is_open( group ) ? 'collapse' : 'expand']( group );\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Open a group by its index within the container (0-based).\r\n\t\t *\r\n\t\t * @param {number} index Zero-based index of the group.\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\topen_by_index(index) {\r\n\t\t\tconst group = this._groups[index];\r\n\t\t\tif ( group ) {\r\n\t\t\t\tthis.expand( group );\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Open a group by matching text contained within the <h3> inside the header.\r\n\t\t * The comparison is case-insensitive and substring-based.\r\n\t\t *\r\n\t\t * @param {string} text Text to match against the heading contents.\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\topen_by_heading(text) {\r\n\t\t\tif ( !text ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tconst t     = String( text ).toLowerCase();\r\n\t\t\tconst match = this._groups.find( (g) => {\r\n\t\t\t\tconst h = g.querySelector( this.opts.header_selector + ' h3' );\r\n\t\t\t\treturn h && h.textContent.toLowerCase().indexOf( t ) !== -1;\r\n\t\t\t} );\r\n\t\t\tif ( match ) {\r\n\t\t\t\tthis.expand( match );\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t// -------------------------------------------------------------------------------------------------------------\r\n\t\t// Internal\r\n\t\t// -------------------------------------------------------------------------------------------------------------\r\n\r\n\t\t/**\r\n\t\t * Delegated click handler for headers.\r\n\t\t *\r\n\t\t * @private\r\n\t\t * @param {MouseEvent} ev The click event.\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\t_on_click(ev) {\r\n\t\t\tconst btn = ev.target.closest( this.opts.header_selector );\r\n\t\t\tif ( !btn || !this.root.contains( btn ) ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tev.preventDefault();\r\n\t\t\tev.stopPropagation();\r\n\t\t\tconst group = btn.closest( this.opts.group_selector );\r\n\t\t\tif ( group ) {\r\n\t\t\t\tthis.toggle( group );\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Keyboard handler for header interactions and roving focus:\r\n\t\t *  - Enter/Space toggles the active group.\r\n\t\t *  - ArrowUp/ArrowDown moves focus between group headers.\r\n\t\t *\r\n\t\t * @private\r\n\t\t * @param {KeyboardEvent} ev The keyboard event.\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\t_on_keydown(ev) {\r\n\t\t\tconst btn = ev.target.closest( this.opts.header_selector );\r\n\t\t\tif ( !btn ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tconst key = ev.key;\r\n\r\n\t\t\t// Toggle on Enter / Space.\r\n\t\t\tif ( key === 'Enter' || key === ' ' ) {\r\n\t\t\t\tev.preventDefault();\r\n\t\t\t\tconst group = btn.closest( this.opts.group_selector );\r\n\t\t\t\tif ( group ) {\r\n\t\t\t\t\tthis.toggle( group );\r\n\t\t\t\t}\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Move focus with ArrowUp/ArrowDown between headers in this container.\r\n\t\t\tif ( key === 'ArrowUp' || key === 'ArrowDown' ) {\r\n\t\t\t\tev.preventDefault();\r\n\t\t\t\tconst headers = Array.prototype.map.call(\r\n\t\t\t\t\tthis.root.querySelectorAll( this.opts.group_selector ),\r\n\t\t\t\t\t(g) => g.querySelector( this.opts.header_selector )\r\n\t\t\t\t).filter( Boolean );\r\n\t\t\t\tconst idx     = headers.indexOf( btn );\r\n\t\t\t\tif ( idx !== -1 ) {\r\n\t\t\t\t\tconst next_idx = (key === 'ArrowDown')\r\n\t\t\t\t\t\t? Math.min( headers.length - 1, idx + 1 )\r\n\t\t\t\t\t\t: Math.max( 0, idx - 1 );\r\n\t\t\t\t\theaders[next_idx].focus();\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Apply ARIA synchronization to all known groups based on their open state.\r\n\t\t *\r\n\t\t * @private\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\t_sync_all_aria() {\r\n\t\t\tthis._groups.forEach( (g) => this._sync_group_aria( g ) );\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Sync ARIA attributes and visibility on a single group.\r\n\t\t *\r\n\t\t * @private\r\n\t\t * @param {HTMLElement} group The group element to sync.\r\n\t\t * @returns {void}\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\t_sync_group_aria(group) {\r\n\t\t\tconst is_open = this.is_open( group );\r\n\t\t\tconst header  = group.querySelector( this.opts.header_selector );\r\n\t\t\tconst panel   = group.querySelector( this.opts.fields_selector );\r\n\r\n\t\t\tif ( header ) {\r\n\t\t\t\t// Header is a real <button>, role is harmless here.\r\n\t\t\t\theader.setAttribute( 'role', 'button' );\r\n\t\t\t\theader.setAttribute( 'aria-expanded', is_open ? 'true' : 'false' );\r\n\r\n\t\t\t\t// Link header to panel by id if available.\r\n\t\t\t\tif ( panel ) {\r\n\t\t\t\t\tif ( !panel.id ) {\r\n\t\t\t\t\t\tpanel.id = this._generate_id( 'wpbc_collapsible_panel' );\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif ( !header.hasAttribute( 'aria-controls' ) ) {\r\n\t\t\t\t\t\theader.setAttribute( 'aria-controls', panel.id );\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t\tif ( panel ) {\r\n\t\t\t\tpanel.hidden = !is_open;\r\n\t\t\t\tpanel.setAttribute( 'aria-hidden', is_open ? 'false' : 'true' );\r\n\t\t\t\t// Optional landmark:\r\n\t\t\t\t// panel.setAttribute('role', 'region');\r\n\t\t\t\t// panel.setAttribute('aria-labelledby', header.id || (header.id = this._generate_id('wpbc_collapsible_header')));\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Internal state change: set a group's open/closed state, sync ARIA,\r\n\t\t * manage focus on collapse, and emit a custom event.\r\n\t\t *\r\n\t\t * @private\r\n\t\t * @param {HTMLElement} group The group element to mutate.\r\n\t\t * @param {boolean} open Whether the group should be open.\r\n\t\t * @returns {void}\r\n\t\t * @fires CustomEvent#wpbc:collapsible:open\r\n\t\t * @fires CustomEvent#wpbc:collapsible:close\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\t_set_open(group, open) {\r\n\t\t\tif ( !open && group.contains( document.activeElement ) ) {\r\n\t\t\t\tconst header = group.querySelector( this.opts.header_selector );\r\n\t\t\t\theader && header.focus();\r\n\t\t\t}\r\n\t\t\tgroup.classList.toggle( this.opts.open_class, open );\r\n\t\t\tthis._sync_group_aria( group );\r\n\t\t\tconst ev_name = open ? 'wpbc:collapsible:open' : 'wpbc:collapsible:close';\r\n\t\t\tgroup.dispatchEvent( new CustomEvent( ev_name, {\r\n\t\t\t\tbubbles: true,\r\n\t\t\t\tdetail : { group, root: this.root, instance: this }\r\n\t\t\t} ) );\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Generate a unique DOM id with the specified prefix.\r\n\t\t *\r\n\t\t * @private\r\n\t\t * @param {string} prefix The id prefix to use.\r\n\t\t * @returns {string} A unique element id not present in the document.\r\n\t\t * @since 2025-08-26\r\n\t\t */\r\n\t\t_generate_id(prefix) {\r\n\t\t\tlet i = 1;\r\n\t\t\tlet id;\r\n\t\t\tdo {\r\n\t\t\t\tid = prefix + '_' + (i++);\r\n\t\t\t}\r\n\t\t\twhile ( d.getElementById( id ) );\r\n\t\t\treturn id;\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Auto-initialize collapsible controllers on the page.\r\n\t * Finds top-level `.wpbc_collapsible` containers (ignoring nested ones),\r\n\t * and instantiates {@link WPBC_Collapsible_Groups} on each.\r\n\t *\r\n\t * @function WPBC_Collapsible_AutoInit\r\n\t * @returns {void}\r\n\t * @since 2025-08-26\r\n\t * @example\r\n\t * // Runs automatically on DOMContentLoaded; can also be called manually:\r\n\t * WPBC_Collapsible_AutoInit();\r\n\t */\r\n\tfunction wpbc_collapsible__auto_init() {\r\n\t\tvar ROOT  = '.wpbc_collapsible';\r\n\t\tvar nodes = Array.prototype.slice.call( d.querySelectorAll( ROOT ) )\r\n\t\t\t.filter( function (n) {\r\n\t\t\t\treturn !n.parentElement || !n.parentElement.closest( ROOT );\r\n\t\t\t} );\r\n\r\n\t\tnodes.forEach( function (node) {\r\n\t\t\tif ( node.__wpbc_collapsible_instance ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tvar exclusive = node.classList.contains( 'wpbc_collapsible--exclusive' ) || node.matches( '[data-wpbc-accordion=\"exclusive\"]' );\r\n\r\n\t\t\tnode.__wpbc_collapsible_instance = new WPBC_Collapsible_Groups( node, { exclusive } ).init();\r\n\t\t} );\r\n\t}\r\n\r\n\t// Export to global for manual control if needed.\r\n\tw.WPBC_Collapsible_Groups   = WPBC_Collapsible_Groups;\r\n\tw.WPBC_Collapsible_AutoInit = wpbc_collapsible__auto_init;\r\n\r\n\t// DOM-ready auto init.\r\n\tif ( d.readyState === 'loading' ) {\r\n\t\td.addEventListener( 'DOMContentLoaded', wpbc_collapsible__auto_init, { once: true } );\r\n\t} else {\r\n\t\twpbc_collapsible__auto_init();\r\n\t}\r\n})( window, document );\r\n"]}
Save
Back