HEX
Server: Apache
System: Linux p3plzcpnl506847.prod.phx3.secureserver.net 4.18.0-553.54.1.lve.el8.x86_64 #1 SMP Wed Jun 4 13:01:13 UTC 2025 x86_64
User: slfopp7cb1df (5698090)
PHP: 8.1.34
Disabled: NONE
Upload Files
File: /home/slfopp7cb1df/www/shaneconrad.me/wp-content/plugins/booking/_dist/all/_src/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,.group__content',
				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 );
			// Only direct children that match.
			const panels = Array.prototype.filter.call( group.children, (el) => el.matches( this.opts.fields_selector ) );

			// Header ARIA.
			if ( header ) {
				header.setAttribute( 'role', 'button' );
				header.setAttribute( 'aria-expanded', is_open ? 'true' : 'false' );

				if ( panels.length ) {
					// Ensure each panel has an id; then wire aria-controls with space-separated ids.
					const ids = panels.map( (p) => {
						if ( !p.id ) p.id = this._generate_id( 'wpbc_collapsible_panel' );
						return p.id;
					} );
					header.setAttribute( 'aria-controls', ids.join( ' ' ) );
				}
			}

			// (3) Panels ARIA + visibility.
			panels.forEach( (p) => {
				p.hidden = !is_open;                            // actual visibility.
				p.setAttribute( 'aria-hidden', is_open ? 'false' : 'true' ); // ARIA.
			} );
		}

		/**
		 * 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 );

/* globals window, document */
/**
 * WPBC Slider Length Groups
 *
 * Universal, dependency-free controller that keeps a "length" control in sync:
 *  - number input  (data-wpbc_slider_len_value)
 *  - unit select   (data-wpbc_slider_len_unit)
 *  - range slider  (data-wpbc_slider_len_range)
 *  - writer input  (data-wpbc_slider_len_writer)  [optional but recommended]
 *
 * The "writer" stores the combined value like: "100%", "420px", "12.5rem".
 * When number/unit/slider change -> writer updates and emits 'input' (bubbles).
 * When writer is changed externally (apply-from-JSON, etc) -> UI updates.
 *
 * Markup expectations (minimal):
 *  <div class="wpbc_slider_len_group"
 *       data-wpbc_slider_len_bounds_map='{"%":{"min":30,"max":100,"step":1},"px":{"min":300,"max":2000,"step":10}}'
 *       data-wpbc_slider_len_default_unit="%">
 *    <input type="number" data-wpbc_slider_len_value>
 *    <select data-wpbc_slider_len_unit>...</select>
 *    <input type="range" data-wpbc_slider_len_range>
 *    <input type="text" data-wpbc_slider_len_writer style="display:none;">
 *  </div>
 *
 * Performance notes:
 * - MutationObserver is DISABLED by default (prevents performance issues).
 * - If your UI re-renders and inserts new groups dynamically, call:
 *     WPBC_Slider_Len_AutoInit();  OR instance.refresh();
 *   Or enable observer via: new WPBC_Slider_Len_Groups(root, { enable_observer:true }).init();
 *
 * Public API (instance methods):
 *  - init(), destroy(), refresh()
 *
 * @version 2026-01-25
 * @since   2026-01-25
 * @file    ../includes/__js/admin/slider_groups/wpbc_len_groups.js
 */
(function (w, d) {
	'use strict';

	// -------------------------------------------------------------------------------------------------
	// Helpers
	// -------------------------------------------------------------------------------------------------
	function clamp_num(v, min, max) {
		if (typeof min === 'number' && !isNaN(min)) v = Math.max(min, v);
		if (typeof max === 'number' && !isNaN(max)) v = Math.min(max, v);
		return v;
	}

	function parse_float(v) {
		var n = parseFloat(v);
		return isNaN(n) ? null : n;
	}

	function safe_json_parse(str) {
		try {
			return JSON.parse(str);
		} catch (e) {
			return null;
		}
	}

	function parse_len_combined(raw, default_unit) {
		var s = (raw == null) ? '' : String(raw).trim();
		if (!s) return { num: '', unit: default_unit || '%' };

		var m = s.match(/^\s*([\-]?\d+(?:\.\d+)?)\s*([a-z%]*)\s*$/i);
		if (!m) {
			// If it's not parseable, treat as number and keep default unit.
			return { num: s, unit: default_unit || '%' };
		}

		var num  = m[1] ? String(m[1]) : '';
		var unit = m[2] ? String(m[2]) : '';
		if (!unit) unit = default_unit || '%';

		return { num: num, unit: unit };
	}

	function build_combined(num, unit) {
		if (num == null || String(num).trim() === '') return '';
		return String(num) + String(unit || '');
	}

	function emit_input(el) {
		if (!el) return;
		el.dispatchEvent(new Event('input', { bubbles: true }));
	}

	// -------------------------------------------------------------------------------------------------
	// Controller
	// -------------------------------------------------------------------------------------------------
	class WPBC_Slider_Len_Groups {

		/**
		 * @param {HTMLElement|string} root_el Container (or selector). If omitted, uses document.
		 * @param {Object} [opts={}]
		 */
		constructor(root_el, opts) {
			this.root = root_el
				? ((typeof root_el === 'string') ? d.querySelector(root_el) : root_el)
				: d;

			this.opts = Object.assign({
				// Strict selectors (NO backward compatibility).
				group_selector  : '.wpbc_slider_len_group',
				value_selector  : '[data-wpbc_slider_len_value]',
				unit_selector   : '[data-wpbc_slider_len_unit]',
				range_selector  : '[data-wpbc_slider_len_range]',
				writer_selector : '[data-wpbc_slider_len_writer]',

				default_unit    : '%',

				fallback_bounds : {
					'px' : { min: 0,   max: 512,  step: 1   },
					'%'  : { min: 0,   max: 100,  step: 1   },
					'rem': { min: 0,   max: 10,   step: 0.1 },
					'em' : { min: 0,   max: 10,   step: 0.1 }
				},

				// Disabled by default for performance.
				enable_observer     : false,
				observer_debounce_ms: 150
			}, opts || {});

			this._on_input  = this._on_input.bind(this);
			this._on_change = this._on_change.bind(this);

			this._bounds_cache = new WeakMap(); // group -> bounds_map_object
			this._observer     = null;
			this._refresh_tmr  = null;
		}

		init() {
			if (!this.root) return this;

			this.root.addEventListener('input',  this._on_input,  true);
			this.root.addEventListener('change', this._on_change, true);

			if (this.opts.enable_observer && w.MutationObserver) {
				this._observer = new MutationObserver(() => { this._debounced_refresh(); });
				this._observer.observe(this.root === d ? d.documentElement : this.root, { childList: true, subtree: true });
			}

			this.refresh();
			return this;
		}

		destroy() {
			if (!this.root) return;

			this.root.removeEventListener('input',  this._on_input,  true);
			this.root.removeEventListener('change', this._on_change, true);

			if (this._observer) {
				this._observer.disconnect();
				this._observer = null;
			}

			if (this._refresh_tmr) {
				clearTimeout(this._refresh_tmr);
				this._refresh_tmr = null;
			}
		}

		refresh() {
			if (!this.root) return;

			var scope  = (this.root === d ? d : this.root);
			var groups = Array.prototype.slice.call(scope.querySelectorAll(this.opts.group_selector));

			for (var i = 0; i < groups.length; i++) {
				this._sync_group_from_writer(groups[i]);
				this._apply_bounds_for_current_unit(groups[i]);
			}
		}

		// -------------------------------------------------------------------------------------------------
		// Internal
		// -------------------------------------------------------------------------------------------------
		_debounced_refresh() {
			if (this._refresh_tmr) clearTimeout(this._refresh_tmr);
			this._refresh_tmr = setTimeout(() => {
				this._refresh_tmr = null;
				this.refresh();
			}, Number(this.opts.observer_debounce_ms) || 0);
		}

		_find_group(el) {
			return (el && el.closest) ? el.closest(this.opts.group_selector) : null;
		}

		_get_parts(group) {
			if (!group) return null;
			return {
				group : group,
				num   : group.querySelector(this.opts.value_selector),
				unit  : group.querySelector(this.opts.unit_selector),
				range : group.querySelector(this.opts.range_selector),
				writer: group.querySelector(this.opts.writer_selector)
			};
		}

		_get_default_unit(group) {
			var du = (group && group.getAttribute)
				? group.getAttribute('data-wpbc_slider_len_default_unit')
				: '';
			return du ? String(du) : this.opts.default_unit;
		}

		_get_bounds_map(group) {
			if (!group) return null;
			if (this._bounds_cache.has(group)) {
				return this._bounds_cache.get(group);
			}

			var raw = group.getAttribute('data-wpbc_slider_len_bounds_map');
			var map = raw ? safe_json_parse(raw) : null;
			if (!map || typeof map !== 'object') map = null;

			this._bounds_cache.set(group, map);
			return map;
		}

		_get_bounds_for_unit(group, unit) {
			var map = this._get_bounds_map(group);
			if (map && unit && map[unit]) {
				return map[unit];
			}
			return this.opts.fallback_bounds[unit] || this.opts.fallback_bounds['px'];
		}

		_apply_bounds(parts, bounds) {
			if (!parts || !bounds) return;

			var min  = (bounds.min  != null) ? Number(bounds.min)  : null;
			var max  = (bounds.max  != null) ? Number(bounds.max)  : null;
			var step = (bounds.step != null) ? Number(bounds.step) : null;

			if (parts.range) {
				if (!isNaN(min))  parts.range.min  = String(min);
				if (!isNaN(max))  parts.range.max  = String(max);
				if (!isNaN(step)) parts.range.step = String(step);
			}
			if (parts.num) {
				if (!isNaN(min))  parts.num.min  = String(min);
				if (!isNaN(max))  parts.num.max  = String(max);
				if (!isNaN(step)) parts.num.step = String(step);
			}
		}

		_apply_bounds_for_current_unit(group) {
			var parts = this._get_parts(group);
			if (!parts || !parts.unit) return;

			var unit = parts.unit.value || this._get_default_unit(group);
			var b    = this._get_bounds_for_unit(group, unit);

			this._apply_bounds(parts, b);

			// Clamp current value to new bounds.
			var v = parse_float(parts.num && parts.num.value ? parts.num.value : (parts.range ? parts.range.value : ''));
			if (v == null) return;

			var min = (b && b.min != null) ? Number(b.min) : null;
			var max = (b && b.max != null) ? Number(b.max) : null;
			v = clamp_num(v, isNaN(min) ? null : min, isNaN(max) ? null : max);

			if (parts.num)   parts.num.value   = String(v);
			if (parts.range) parts.range.value = String(v);

			this._write_combined(parts, String(v), unit, /*emit*/ false);
		}

		_write_combined(parts, num, unit, emit) {
			if (!parts) return;

			var combined = build_combined(num, unit);

			if (parts.writer) {
				// Avoid recursion: mark as internal write.
				parts.writer.__wpbc_slider_len_internal = true;
				parts.writer.value = combined;
				if (emit) emit_input(parts.writer);
				parts.writer.__wpbc_slider_len_internal = false;
			} else if (parts.num) {
				// If writer is missing, at least notify via number input.
				if (emit) emit_input(parts.num);
			}
		}

		_sync_group_from_writer(group) {
			var parts = this._get_parts(group);
			if (!parts || !parts.writer) return;

			var raw = String(parts.writer.value || '').trim();
			if (!raw) return;

			var du = this._get_default_unit(group);
			var p  = parse_len_combined(raw, du);

			if (parts.unit)  parts.unit.value  = p.unit;
			if (parts.num)   parts.num.value   = p.num;
			if (parts.range) parts.range.value = p.num;
		}

		_on_input(ev) {
			var t = ev.target;
			if (!t) return;

			var group = this._find_group(t);
			if (!group) return;

			var parts = this._get_parts(group);
			if (!parts) return;

			// Writer changed externally -> update UI.
			if (parts.writer && t === parts.writer) {
				if (t.__wpbc_slider_len_internal) return;
				this._sync_group_from_writer(group);
				this._apply_bounds_for_current_unit(group);
				return;
			}

			// Slider moved -> update number + writer.
			if (t.matches && t.matches(this.opts.range_selector)) {
				if (parts.num) parts.num.value = t.value;

				var unit = (parts.unit && parts.unit.value) ? parts.unit.value : this._get_default_unit(group);
				this._write_combined(parts, t.value, unit, /*emit*/ true);
				return;
			}

			// Number typed -> update slider + writer (clamp if slider has bounds).
			if (t.matches && t.matches(this.opts.value_selector)) {
				var v = parse_float(t.value);

				if (v != null && parts.range) {
					var rmin = Number(parts.range.min);
					var rmax = Number(parts.range.max);
					v = clamp_num(v, isNaN(rmin) ? null : rmin, isNaN(rmax) ? null : rmax);

					parts.range.value = String(v);
					if (String(v) !== t.value) t.value = String(v);
				}

				var unit2 = (parts.unit && parts.unit.value) ? parts.unit.value : this._get_default_unit(group);
				this._write_combined(parts, t.value, unit2, /*emit*/ true);
			}
		}

		_on_change(ev) {
			var t = ev.target;
			if (!t) return;

			var group = this._find_group(t);
			if (!group) return;

			var parts = this._get_parts(group);
			if (!parts) return;

			// Unit changed -> update bounds + writer.
			if (t.matches && t.matches(this.opts.unit_selector)) {
				this._apply_bounds_for_current_unit(group);

				var num  = parts.num ? parts.num.value : (parts.range ? parts.range.value : '');
				var unit = t.value || this._get_default_unit(group);
				this._write_combined(parts, num, unit, /*emit*/ true);
			}
		}
	}

	// -------------------------------------------------------------------------------------------------
	// Auto-init
	// -------------------------------------------------------------------------------------------------
	function wpbc_slider_len_groups__auto_init() {
		var ROOT  = '.wpbc_slider_len_groups';
		var nodes = Array.prototype.slice.call(d.querySelectorAll(ROOT))
			.filter(function (n) { return !n.parentElement || !n.parentElement.closest(ROOT); });

		// If no explicit containers, install a single document-root instance.
		if (!nodes.length) {
			if (!d.__wpbc_slider_len_groups_global_instance) {
				d.__wpbc_slider_len_groups_global_instance = new WPBC_Slider_Len_Groups(d).init();
			}
			return;
		}

		nodes.forEach(function (node) {
			if (node.__wpbc_slider_len_groups_instance) return;
			node.__wpbc_slider_len_groups_instance = new WPBC_Slider_Len_Groups(node).init();
		});
	}

	// Export globals (manual control if needed).
	w.WPBC_Slider_Len_Groups   = WPBC_Slider_Len_Groups;
	w.WPBC_Slider_Len_AutoInit = wpbc_slider_len_groups__auto_init;

	// DOM-ready auto init.
	if (d.readyState === 'loading') {
		d.addEventListener('DOMContentLoaded', wpbc_slider_len_groups__auto_init, { once: true });
	} else {
		wpbc_slider_len_groups__auto_init();
	}

})(window, document);

/* globals window, document */
/**
 * WPBC Slider Range Groups
 *
 * Universal, dependency-free controller that keeps a "range + number" pair in sync:
 *  - number input  (data-wpbc_slider_range_value)
 *  - range slider  (data-wpbc_slider_range_range)
 *  - writer input  (data-wpbc_slider_range_writer) [optional]
 *
 * If writer exists: number/slider update writer and emit 'input' on writer (bubbles).
 * If writer is missing: emits 'input' on the number input.
 * If writer changes externally: updates number/slider.
 *
 * Markup expectations (minimal):
 *  <div class="wpbc_slider_range_group">
 *    <input type="number" data-wpbc_slider_range_value>
 *    <input type="range"  data-wpbc_slider_range_range>
 *    <!-- optional -->
 *    <input type="text" data-wpbc_slider_range_writer style="display:none;">
 *  </div>
 *
 * Performance notes:
 * - MutationObserver is DISABLED by default.
 * - If your UI re-renders and inserts new groups dynamically, call:
 *     WPBC_Slider_Range_AutoInit(); OR instance.refresh();
 *   Or enable observer via: new WPBC_Slider_Range_Groups(root, { enable_observer:true }).init();
 *
 * Public API (instance methods):
 *  - init(), destroy(), refresh()
 *
 * @version 2026-01-25
 * @since   2026-01-25
 * @file    ../includes/__js/admin/slider_groups/wpbc_range_groups.js
 */
(function (w, d) {
	'use strict';

	// -------------------------------------------------------------------------------------------------
	// Helpers
	// -------------------------------------------------------------------------------------------------
	function clamp_num(v, min, max) {
		if (typeof min === 'number' && !isNaN(min)) v = Math.max(min, v);
		if (typeof max === 'number' && !isNaN(max)) v = Math.min(max, v);
		return v;
	}

	function parse_float(v) {
		var n = parseFloat(v);
		return isNaN(n) ? null : n;
	}

	function emit_input(el) {
		if (!el) return;
		el.dispatchEvent(new Event('input', { bubbles: true }));
	}

	// -------------------------------------------------------------------------------------------------
	// Controller
	// -------------------------------------------------------------------------------------------------
	class WPBC_Slider_Range_Groups {

		/**
		 * @param {HTMLElement|string} root_el Container (or selector). If omitted, uses document.
		 * @param {Object} [opts={}]
		 */
		constructor(root_el, opts) {
			this.root = root_el
				? ((typeof root_el === 'string') ? d.querySelector(root_el) : root_el)
				: d;

			this.opts = Object.assign({
				// Strict selectors (NO backward compatibility).
				group_selector  : '.wpbc_slider_range_group',
				value_selector  : '[data-wpbc_slider_range_value]',
				range_selector  : '[data-wpbc_slider_range_range]',
				writer_selector : '[data-wpbc_slider_range_writer]',

				// Disabled by default for performance.
				enable_observer     : false,
				observer_debounce_ms: 150
			}, opts || {});

			this._on_input  = this._on_input.bind(this);
			this._on_change = this._on_change.bind(this);

			this._observer    = null;
			this._refresh_tmr = null;
		}

		init() {
			if (!this.root) return this;

			this.root.addEventListener('input',  this._on_input,  true);
			this.root.addEventListener('change', this._on_change, true);

			if (this.opts.enable_observer && w.MutationObserver) {
				this._observer = new MutationObserver(() => { this._debounced_refresh(); });
				this._observer.observe(this.root === d ? d.documentElement : this.root, { childList: true, subtree: true });
			}

			this.refresh();
			return this;
		}

		destroy() {
			if (!this.root) return;

			this.root.removeEventListener('input',  this._on_input,  true);
			this.root.removeEventListener('change', this._on_change, true);

			if (this._observer) {
				this._observer.disconnect();
				this._observer = null;
			}

			if (this._refresh_tmr) {
				clearTimeout(this._refresh_tmr);
				this._refresh_tmr = null;
			}
		}

		refresh() {
			if (!this.root) return;

			var scope  = (this.root === d ? d : this.root);
			var groups = Array.prototype.slice.call(scope.querySelectorAll(this.opts.group_selector));

			for (var i = 0; i < groups.length; i++) {
				this._sync_from_writer(groups[i]);
				this._clamp_to_range(groups[i]);
			}
		}

		// -------------------------------------------------------------------------------------------------
		// Internal
		// -------------------------------------------------------------------------------------------------
		_debounced_refresh() {
			if (this._refresh_tmr) clearTimeout(this._refresh_tmr);
			this._refresh_tmr = setTimeout(() => {
				this._refresh_tmr = null;
				this.refresh();
			}, Number(this.opts.observer_debounce_ms) || 0);
		}

		_find_group(el) {
			return (el && el.closest) ? el.closest(this.opts.group_selector) : null;
		}

		_get_parts(group) {
			if (!group) return null;
			return {
				group : group,
				num   : group.querySelector(this.opts.value_selector),
				range : group.querySelector(this.opts.range_selector),
				writer: group.querySelector(this.opts.writer_selector)
			};
		}

		_write(parts, value, emit) {
			if (!parts) return;

			if (parts.writer) {
				parts.writer.__wpbc_slider_range_internal = true;
				parts.writer.value = String(value);
				if (emit) emit_input(parts.writer);
				parts.writer.__wpbc_slider_range_internal = false;
			} else if (parts.num) {
				// If writer is missing, at least notify via number input.
				if (emit) emit_input(parts.num);
			}
		}

		_sync_from_writer(group) {
			var parts = this._get_parts(group);
			if (!parts || !parts.writer) return;

			var raw = String(parts.writer.value || '').trim();
			if (!raw) return;

			if (parts.num)   parts.num.value   = raw;
			if (parts.range) parts.range.value = raw;
		}

		_clamp_to_range(group) {
			var parts = this._get_parts(group);
			if (!parts || !parts.range || !parts.num) return;

			var v = parse_float(parts.num.value);
			if (v == null) return;

			var min = Number(parts.range.min);
			var max = Number(parts.range.max);
			var vv  = clamp_num(v, isNaN(min) ? null : min, isNaN(max) ? null : max);

			if (String(vv) !== parts.num.value) parts.num.value = String(vv);
			parts.range.value = String(vv);
		}

		_on_input(ev) {
			var t = ev.target;
			if (!t) return;

			var group = this._find_group(t);
			if (!group) return;

			var parts = this._get_parts(group);
			if (!parts) return;

			// Writer changed externally -> update UI.
			if (parts.writer && t === parts.writer) {
				if (t.__wpbc_slider_range_internal) return;
				this._sync_from_writer(group);
				this._clamp_to_range(group);
				return;
			}

			// Range moved -> update number + writer.
			if (t.matches && t.matches(this.opts.range_selector)) {
				if (parts.num) parts.num.value = t.value;
				this._write(parts, t.value, /*emit*/ true);
				return;
			}

			// Number typed -> update range + writer (clamp by slider bounds).
			if (t.matches && t.matches(this.opts.value_selector)) {
				if (parts.range) {
					var v = parse_float(t.value);
					if (v != null) {
						var min = Number(parts.range.min);
						var max = Number(parts.range.max);
						v = clamp_num(v, isNaN(min) ? null : min, isNaN(max) ? null : max);

						parts.range.value = String(v);
						if (String(v) !== t.value) t.value = String(v);
					}
				}
				this._write(parts, t.value, /*emit*/ true);
			}
		}

		_on_change(ev) {
			// No special "change" handling needed currently; kept for symmetry/future.
		}
	}

	// -------------------------------------------------------------------------------------------------
	// Auto-init
	// -------------------------------------------------------------------------------------------------
	function wpbc_slider_range_groups__auto_init() {
		var ROOT  = '.wpbc_slider_range_groups';
		var nodes = Array.prototype.slice.call(d.querySelectorAll(ROOT))
			.filter(function (n) { return !n.parentElement || !n.parentElement.closest(ROOT); });

		if (!nodes.length) {
			if (!d.__wpbc_slider_range_groups_global_instance) {
				d.__wpbc_slider_range_groups_global_instance = new WPBC_Slider_Range_Groups(d).init();
			}
			return;
		}

		nodes.forEach(function (node) {
			if (node.__wpbc_slider_range_groups_instance) return;
			node.__wpbc_slider_range_groups_instance = new WPBC_Slider_Range_Groups(node).init();
		});
	}

	// Export globals.
	w.WPBC_Slider_Range_Groups   = WPBC_Slider_Range_Groups;
	w.WPBC_Slider_Range_AutoInit = wpbc_slider_range_groups__auto_init;

	if (d.readyState === 'loading') {
		d.addEventListener('DOMContentLoaded', wpbc_slider_range_groups__auto_init, { once: true });
	} else {
		wpbc_slider_range_groups__auto_init();
	}

})(window, document);

/**
 * Booking Calendar — Generic UI Tabs Utility (JS)
 *
 * Purpose: Lightweight, dependency-free tabs controller for any small tab group in admin UIs.
 * - Auto-initializes groups marked with data-wpbc-tabs.
 * - Assigns ARIA roles and toggles aria-selected/aria-hidden/tabindex.
 * - Supports keyboard navigation (Left/Right/Home/End).
 * - Public API: window.wpbc_ui_tabs.{init_on, init_group, set_active}
 * - Emits 'wpbc:tabs:change' on the group root when the active tab changes.
 *
 * Markup contract:
 * - Root:   [data-wpbc-tabs]
 * - Tabs:   [data-wpbc-tab-key="K"]
 * - Panels: [data-wpbc-tab-panel="K"]
 *
 * @package   Booking Calendar
 * @subpackage Admin\UI
 * @since     11.0.0
 * @version   1.0.0
 * @see       /includes/__js/admin/ui_tabs/ui_tabs.js
 *
 *
 * How it works:
 * - Root node must have [data-wpbc-tabs] attribute (any value).
 * - Tab buttons must carry [data-wpbc-tab-key="..."] (unique per group).
 * - Panels must carry [data-wpbc-tab-panel="..."] with matching keys.
 * - Adds WAI-ARIA roles and aria-selected/hidden wiring.
 *
 * <div data-wpbc-tabs="column-styles" data-wpbc-tab-active="1"    class="wpbc_ui_tabs_root" >
 *    <!-- Top Tabs -->
 *    <div data-wpbc-tablist="" role="tablist"                    class=" wpbc_ui_el__horis_top_bar__wrapper" >
 *        <div class="wpbc_ui_el__horis_top_bar__content">
 *            <h2 class="wpbc_ui_el__horis_nav_label">Column:</h2>
 *
 *            <div class="wpbc_ui_el__horis_nav_item wpbc_ui_el__horis_nav_item__1">
 *                <a
 *                    data-wpbc-tab-key="1"
 *                    aria-selected="true" role="tab" tabindex="0" aria-controls="wpbc_tab_panel_col_1"
 *
 *                        href="javascript:void(0);"
 *                        class="wpbc_ui_el__horis_nav_item__a wpbc_ui_el__horis_nav_item__single"
 *                        id="wpbc_tab_col_1"
 *                        title="Column 1"
 *                ><span class="wpbc_ui_el__horis_nav_title">Title 1</span></a>
 *            </div>
 *            ...
 *        </div>
 *    </div>
 *    <!-- Tabs Content -->
 *    <div class="wpbc_tab__panel group__fields" data-wpbc-tab-panel="1" id="wpbc_tab_panel_col_1" role="tabpanel" aria-labelledby="wpbc_tab_col_1">
 *        ...
 *    </div>
 *    ...
 * </div>
 *
 * Public API:
 *   - wpbc_ui_tabs.init_on(root_or_selector)   // find and init groups within a container
 *   - wpbc_ui_tabs.init_group(root_el)         // init a single group root
 *   - wpbc_ui_tabs.set_active(root_el, key)    // programmatically change active tab
 *
 * Events:
 *   - Dispatches CustomEvent 'wpbc:tabs:change' on root when tab changes:
 *       detail: { active_key: '2', prev_key: '1' }
 *
 * Switch a local (generic) tabs group to tab 3:     var group = document.querySelector('[data-wpbc-tabs="column-styles"]'); if ( group ) { wpbc_ui_tabs.set_active(group, '3'); }
 */
(function ( w ) {
	'use strict';

	if ( w.wpbc_ui_tabs ) {
		return;
	}

	/**
	 * Internal: toggle active state.
	 *
	 * @param {HTMLElement} root_el
	 * @param {string}      key
	 * @param {boolean}     should_emit
	 */
	function set_active_internal( root_el, key, should_emit ) {
		var tab_btns = root_el.querySelectorAll( '[data-wpbc-tab-key]' );
		var panels   = root_el.querySelectorAll( '[data-wpbc-tab-panel]' );

		var prev_key = root_el.getAttribute( 'data-wpbc-tab-active' ) || null;
		if ( String( prev_key ) === String( key ) ) {
			return;
		}

		// Buttons: aria + class
		for ( var i = 0; i < tab_btns.length; i++ ) {
			var btn   = tab_btns[i];
			var b_key = btn.getAttribute( 'data-wpbc-tab-key' );
			var is_on = String( b_key ) === String( key );

			btn.setAttribute( 'role', 'tab' );
			btn.setAttribute( 'aria-selected', is_on ? 'true' : 'false' );
			btn.setAttribute( 'tabindex', is_on ? '0' : '-1' );

			if ( is_on ) {
				btn.classList.add( 'active' );
			} else {
				btn.classList.remove( 'active' );
			}
		}

		// Panels: aria + visibility
		for ( var j = 0; j < panels.length; j++ ) {
			var pn   = panels[j];
			var pkey = pn.getAttribute( 'data-wpbc-tab-panel' );
			var show = String( pkey ) === String( key );

			pn.setAttribute( 'role', 'tabpanel' );
			pn.setAttribute( 'aria-hidden', show ? 'false' : 'true' );
			if ( show ) {
				pn.removeAttribute( 'hidden' );
			} else {
				pn.setAttribute( 'hidden', '' );
			}
		}

		root_el.setAttribute( 'data-wpbc-tab-active', String( key ) );

		if ( should_emit ) {
			try {
				var ev = new w.CustomEvent( 'wpbc:tabs:change', {
					bubbles : true,
					detail  : { active_key : String( key ), prev_key : prev_key }
				} );
				root_el.dispatchEvent( ev );
			} catch ( _e ) {}
		}
	}

	/**
	 * Internal: get ordered keys from buttons.
	 *
	 * @param {HTMLElement} root_el
	 * @returns {string[]}
	 */
	function get_keys( root_el ) {
		var list = [];
		var btns = root_el.querySelectorAll( '[data-wpbc-tab-key]' );
		for ( var i = 0; i < btns.length; i++ ) {
			var k = btns[i].getAttribute( 'data-wpbc-tab-key' );
			if ( k != null && k !== '' ) {
				list.push( String( k ) );
			}
		}
		return list;
	}

	/**
	 * Internal: move focus between tabs using keyboard.
	 *
	 * @param {HTMLElement} root_el
	 * @param {number}      dir  +1 (next) / -1 (prev)
	 */
	function focus_relative( root_el, dir ) {
		var keys    = get_keys( root_el );
		var current = root_el.getAttribute( 'data-wpbc-tab-active' ) || keys[0] || null;
		var idx     = Math.max( 0, keys.indexOf( String( current ) ) );
		var next    = keys[ ( idx + ( dir > 0 ? 1 : keys.length - 1 ) ) % keys.length ];

		var next_btn = root_el.querySelector( '[data-wpbc-tab-key="' + next + '"]' );
		if ( next_btn ) {
			next_btn.focus();
			set_active_internal( root_el, next, true );
		}
	}

	/**
	 * Initialize a single tabs group root.
	 *
	 * @param {HTMLElement} root_el
	 */
	function init_group( root_el ) {
		if ( ! root_el || root_el.__wpbc_tabs_inited ) {
			return;
		}
		root_el.__wpbc_tabs_inited = true;

		// Roles
		var tablist = root_el.querySelector( '[data-wpbc-tablist]' ) || root_el;
		tablist.setAttribute( 'role', 'tablist' );

		// Default active: from attribute or first button
		var keys = get_keys( root_el );
		var def  = root_el.getAttribute( 'data-wpbc-tab-active' ) || ( keys[0] || '1' );
		set_active_internal( root_el, def, false );

		// Clicks
		root_el.addEventListener( 'click', function ( e ) {
			var btn = e.target.closest ? e.target.closest( '[data-wpbc-tab-key]' ) : null;
			if ( ! btn || ! root_el.contains( btn ) ) {
				return;
			}
			e.preventDefault();
			var key = btn.getAttribute( 'data-wpbc-tab-key' );
			if ( key != null ) {
				set_active_internal( root_el, key, true );
			}
		}, true );

		// Keyboard (Left/Right/Home/End)
		root_el.addEventListener( 'keydown', function ( e ) {
			var tgt = e.target;
			if ( ! tgt || ! tgt.hasAttribute || ! tgt.hasAttribute( 'data-wpbc-tab-key' ) ) {
				return;
			}
			switch ( e.key ) {
			case 'ArrowLeft':
				e.preventDefault(); focus_relative( root_el, -1 ); break;
			case 'ArrowRight':
				e.preventDefault(); focus_relative( root_el, +1 ); break;
			case 'Home':
				e.preventDefault(); set_active_internal( root_el, ( get_keys( root_el )[0] || '1' ), true ); break;
			case 'End':
				e.preventDefault(); var ks = get_keys( root_el ); set_active_internal( root_el, ( ks[ ks.length - 1 ] || '1' ), true ); break;
			}
		}, true );
	}

	/**
	 * Initialize all groups within a container (or document).
	 *
	 * @param {HTMLElement|string|null} container
	 */
	function init_on( container ) {
		var ctx = container ? ( typeof container === 'string' ? document.querySelector( container ) : container ) : document;
		if ( ! ctx ) {
			return;
		}
		var groups = ctx.querySelectorAll( '[data-wpbc-tabs]' );
		for ( var i = 0; i < groups.length; i++ ) {
			init_group( groups[i] );
		}
	}

	/**
	 * Programmatically set active tab by key.
	 *
	 * @param {HTMLElement} root_el
	 * @param {string|number} key
	 */
	function set_active( root_el, key ) {
		if ( root_el && root_el.hasAttribute && root_el.hasAttribute( 'data-wpbc-tabs' ) ) {
			set_active_internal( root_el, String( key ), true );
		}
	}

	// Public API (snake_case)
	w.wpbc_ui_tabs = {
		init_on    : init_on,
		init_group : init_group,
		set_active : set_active
	};

	// Auto-init on DOM ready
	if ( document.readyState === 'loading' ) {
		document.addEventListener( 'DOMContentLoaded', function () { init_on( document ); } );
	} else {
		init_on( document );
	}

})( window );

//# 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","wpbc_len_groups.js","wpbc_range_groups.js","ui_tabs.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;AClgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACtZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACrRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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,.group__content',\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\t// Only direct children that match.\r\n\t\t\tconst panels = Array.prototype.filter.call( group.children, (el) => el.matches( this.opts.fields_selector ) );\r\n\r\n\t\t\t// Header ARIA.\r\n\t\t\tif ( header ) {\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\tif ( panels.length ) {\r\n\t\t\t\t\t// Ensure each panel has an id; then wire aria-controls with space-separated ids.\r\n\t\t\t\t\tconst ids = panels.map( (p) => {\r\n\t\t\t\t\t\tif ( !p.id ) p.id = this._generate_id( 'wpbc_collapsible_panel' );\r\n\t\t\t\t\t\treturn p.id;\r\n\t\t\t\t\t} );\r\n\t\t\t\t\theader.setAttribute( 'aria-controls', ids.join( ' ' ) );\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\t// (3) Panels ARIA + visibility.\r\n\t\t\tpanels.forEach( (p) => {\r\n\t\t\t\tp.hidden = !is_open;                            // actual visibility.\r\n\t\t\t\tp.setAttribute( 'aria-hidden', is_open ? 'false' : 'true' ); // ARIA.\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","/* globals window, document */\r\n/**\r\n * WPBC Slider Length Groups\r\n *\r\n * Universal, dependency-free controller that keeps a \"length\" control in sync:\r\n *  - number input  (data-wpbc_slider_len_value)\r\n *  - unit select   (data-wpbc_slider_len_unit)\r\n *  - range slider  (data-wpbc_slider_len_range)\r\n *  - writer input  (data-wpbc_slider_len_writer)  [optional but recommended]\r\n *\r\n * The \"writer\" stores the combined value like: \"100%\", \"420px\", \"12.5rem\".\r\n * When number/unit/slider change -> writer updates and emits 'input' (bubbles).\r\n * When writer is changed externally (apply-from-JSON, etc) -> UI updates.\r\n *\r\n * Markup expectations (minimal):\r\n *  <div class=\"wpbc_slider_len_group\"\r\n *       data-wpbc_slider_len_bounds_map='{\"%\":{\"min\":30,\"max\":100,\"step\":1},\"px\":{\"min\":300,\"max\":2000,\"step\":10}}'\r\n *       data-wpbc_slider_len_default_unit=\"%\">\r\n *    <input type=\"number\" data-wpbc_slider_len_value>\r\n *    <select data-wpbc_slider_len_unit>...</select>\r\n *    <input type=\"range\" data-wpbc_slider_len_range>\r\n *    <input type=\"text\" data-wpbc_slider_len_writer style=\"display:none;\">\r\n *  </div>\r\n *\r\n * Performance notes:\r\n * - MutationObserver is DISABLED by default (prevents performance issues).\r\n * - If your UI re-renders and inserts new groups dynamically, call:\r\n *     WPBC_Slider_Len_AutoInit();  OR instance.refresh();\r\n *   Or enable observer via: new WPBC_Slider_Len_Groups(root, { enable_observer:true }).init();\r\n *\r\n * Public API (instance methods):\r\n *  - init(), destroy(), refresh()\r\n *\r\n * @version 2026-01-25\r\n * @since   2026-01-25\r\n * @file    ../includes/__js/admin/slider_groups/wpbc_len_groups.js\r\n */\r\n(function (w, d) {\r\n\t'use strict';\r\n\r\n\t// -------------------------------------------------------------------------------------------------\r\n\t// Helpers\r\n\t// -------------------------------------------------------------------------------------------------\r\n\tfunction clamp_num(v, min, max) {\r\n\t\tif (typeof min === 'number' && !isNaN(min)) v = Math.max(min, v);\r\n\t\tif (typeof max === 'number' && !isNaN(max)) v = Math.min(max, v);\r\n\t\treturn v;\r\n\t}\r\n\r\n\tfunction parse_float(v) {\r\n\t\tvar n = parseFloat(v);\r\n\t\treturn isNaN(n) ? null : n;\r\n\t}\r\n\r\n\tfunction safe_json_parse(str) {\r\n\t\ttry {\r\n\t\t\treturn JSON.parse(str);\r\n\t\t} catch (e) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\t}\r\n\r\n\tfunction parse_len_combined(raw, default_unit) {\r\n\t\tvar s = (raw == null) ? '' : String(raw).trim();\r\n\t\tif (!s) return { num: '', unit: default_unit || '%' };\r\n\r\n\t\tvar m = s.match(/^\\s*([\\-]?\\d+(?:\\.\\d+)?)\\s*([a-z%]*)\\s*$/i);\r\n\t\tif (!m) {\r\n\t\t\t// If it's not parseable, treat as number and keep default unit.\r\n\t\t\treturn { num: s, unit: default_unit || '%' };\r\n\t\t}\r\n\r\n\t\tvar num  = m[1] ? String(m[1]) : '';\r\n\t\tvar unit = m[2] ? String(m[2]) : '';\r\n\t\tif (!unit) unit = default_unit || '%';\r\n\r\n\t\treturn { num: num, unit: unit };\r\n\t}\r\n\r\n\tfunction build_combined(num, unit) {\r\n\t\tif (num == null || String(num).trim() === '') return '';\r\n\t\treturn String(num) + String(unit || '');\r\n\t}\r\n\r\n\tfunction emit_input(el) {\r\n\t\tif (!el) return;\r\n\t\tel.dispatchEvent(new Event('input', { bubbles: true }));\r\n\t}\r\n\r\n\t// -------------------------------------------------------------------------------------------------\r\n\t// Controller\r\n\t// -------------------------------------------------------------------------------------------------\r\n\tclass WPBC_Slider_Len_Groups {\r\n\r\n\t\t/**\r\n\t\t * @param {HTMLElement|string} root_el Container (or selector). If omitted, uses document.\r\n\t\t * @param {Object} [opts={}]\r\n\t\t */\r\n\t\tconstructor(root_el, opts) {\r\n\t\t\tthis.root = root_el\r\n\t\t\t\t? ((typeof root_el === 'string') ? d.querySelector(root_el) : root_el)\r\n\t\t\t\t: d;\r\n\r\n\t\t\tthis.opts = Object.assign({\r\n\t\t\t\t// Strict selectors (NO backward compatibility).\r\n\t\t\t\tgroup_selector  : '.wpbc_slider_len_group',\r\n\t\t\t\tvalue_selector  : '[data-wpbc_slider_len_value]',\r\n\t\t\t\tunit_selector   : '[data-wpbc_slider_len_unit]',\r\n\t\t\t\trange_selector  : '[data-wpbc_slider_len_range]',\r\n\t\t\t\twriter_selector : '[data-wpbc_slider_len_writer]',\r\n\r\n\t\t\t\tdefault_unit    : '%',\r\n\r\n\t\t\t\tfallback_bounds : {\r\n\t\t\t\t\t'px' : { min: 0,   max: 512,  step: 1   },\r\n\t\t\t\t\t'%'  : { min: 0,   max: 100,  step: 1   },\r\n\t\t\t\t\t'rem': { min: 0,   max: 10,   step: 0.1 },\r\n\t\t\t\t\t'em' : { min: 0,   max: 10,   step: 0.1 }\r\n\t\t\t\t},\r\n\r\n\t\t\t\t// Disabled by default for performance.\r\n\t\t\t\tenable_observer     : false,\r\n\t\t\t\tobserver_debounce_ms: 150\r\n\t\t\t}, opts || {});\r\n\r\n\t\t\tthis._on_input  = this._on_input.bind(this);\r\n\t\t\tthis._on_change = this._on_change.bind(this);\r\n\r\n\t\t\tthis._bounds_cache = new WeakMap(); // group -> bounds_map_object\r\n\t\t\tthis._observer     = null;\r\n\t\t\tthis._refresh_tmr  = null;\r\n\t\t}\r\n\r\n\t\tinit() {\r\n\t\t\tif (!this.root) return this;\r\n\r\n\t\t\tthis.root.addEventListener('input',  this._on_input,  true);\r\n\t\t\tthis.root.addEventListener('change', this._on_change, true);\r\n\r\n\t\t\tif (this.opts.enable_observer && w.MutationObserver) {\r\n\t\t\t\tthis._observer = new MutationObserver(() => { this._debounced_refresh(); });\r\n\t\t\t\tthis._observer.observe(this.root === d ? d.documentElement : this.root, { childList: true, subtree: true });\r\n\t\t\t}\r\n\r\n\t\t\tthis.refresh();\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\tdestroy() {\r\n\t\t\tif (!this.root) return;\r\n\r\n\t\t\tthis.root.removeEventListener('input',  this._on_input,  true);\r\n\t\t\tthis.root.removeEventListener('change', this._on_change, true);\r\n\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\r\n\t\t\tif (this._refresh_tmr) {\r\n\t\t\t\tclearTimeout(this._refresh_tmr);\r\n\t\t\t\tthis._refresh_tmr = null;\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\trefresh() {\r\n\t\t\tif (!this.root) return;\r\n\r\n\t\t\tvar scope  = (this.root === d ? d : this.root);\r\n\t\t\tvar groups = Array.prototype.slice.call(scope.querySelectorAll(this.opts.group_selector));\r\n\r\n\t\t\tfor (var i = 0; i < groups.length; i++) {\r\n\t\t\t\tthis._sync_group_from_writer(groups[i]);\r\n\t\t\t\tthis._apply_bounds_for_current_unit(groups[i]);\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\t\t_debounced_refresh() {\r\n\t\t\tif (this._refresh_tmr) clearTimeout(this._refresh_tmr);\r\n\t\t\tthis._refresh_tmr = setTimeout(() => {\r\n\t\t\t\tthis._refresh_tmr = null;\r\n\t\t\t\tthis.refresh();\r\n\t\t\t}, Number(this.opts.observer_debounce_ms) || 0);\r\n\t\t}\r\n\r\n\t\t_find_group(el) {\r\n\t\t\treturn (el && el.closest) ? el.closest(this.opts.group_selector) : null;\r\n\t\t}\r\n\r\n\t\t_get_parts(group) {\r\n\t\t\tif (!group) return null;\r\n\t\t\treturn {\r\n\t\t\t\tgroup : group,\r\n\t\t\t\tnum   : group.querySelector(this.opts.value_selector),\r\n\t\t\t\tunit  : group.querySelector(this.opts.unit_selector),\r\n\t\t\t\trange : group.querySelector(this.opts.range_selector),\r\n\t\t\t\twriter: group.querySelector(this.opts.writer_selector)\r\n\t\t\t};\r\n\t\t}\r\n\r\n\t\t_get_default_unit(group) {\r\n\t\t\tvar du = (group && group.getAttribute)\r\n\t\t\t\t? group.getAttribute('data-wpbc_slider_len_default_unit')\r\n\t\t\t\t: '';\r\n\t\t\treturn du ? String(du) : this.opts.default_unit;\r\n\t\t}\r\n\r\n\t\t_get_bounds_map(group) {\r\n\t\t\tif (!group) return null;\r\n\t\t\tif (this._bounds_cache.has(group)) {\r\n\t\t\t\treturn this._bounds_cache.get(group);\r\n\t\t\t}\r\n\r\n\t\t\tvar raw = group.getAttribute('data-wpbc_slider_len_bounds_map');\r\n\t\t\tvar map = raw ? safe_json_parse(raw) : null;\r\n\t\t\tif (!map || typeof map !== 'object') map = null;\r\n\r\n\t\t\tthis._bounds_cache.set(group, map);\r\n\t\t\treturn map;\r\n\t\t}\r\n\r\n\t\t_get_bounds_for_unit(group, unit) {\r\n\t\t\tvar map = this._get_bounds_map(group);\r\n\t\t\tif (map && unit && map[unit]) {\r\n\t\t\t\treturn map[unit];\r\n\t\t\t}\r\n\t\t\treturn this.opts.fallback_bounds[unit] || this.opts.fallback_bounds['px'];\r\n\t\t}\r\n\r\n\t\t_apply_bounds(parts, bounds) {\r\n\t\t\tif (!parts || !bounds) return;\r\n\r\n\t\t\tvar min  = (bounds.min  != null) ? Number(bounds.min)  : null;\r\n\t\t\tvar max  = (bounds.max  != null) ? Number(bounds.max)  : null;\r\n\t\t\tvar step = (bounds.step != null) ? Number(bounds.step) : null;\r\n\r\n\t\t\tif (parts.range) {\r\n\t\t\t\tif (!isNaN(min))  parts.range.min  = String(min);\r\n\t\t\t\tif (!isNaN(max))  parts.range.max  = String(max);\r\n\t\t\t\tif (!isNaN(step)) parts.range.step = String(step);\r\n\t\t\t}\r\n\t\t\tif (parts.num) {\r\n\t\t\t\tif (!isNaN(min))  parts.num.min  = String(min);\r\n\t\t\t\tif (!isNaN(max))  parts.num.max  = String(max);\r\n\t\t\t\tif (!isNaN(step)) parts.num.step = String(step);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t_apply_bounds_for_current_unit(group) {\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts || !parts.unit) return;\r\n\r\n\t\t\tvar unit = parts.unit.value || this._get_default_unit(group);\r\n\t\t\tvar b    = this._get_bounds_for_unit(group, unit);\r\n\r\n\t\t\tthis._apply_bounds(parts, b);\r\n\r\n\t\t\t// Clamp current value to new bounds.\r\n\t\t\tvar v = parse_float(parts.num && parts.num.value ? parts.num.value : (parts.range ? parts.range.value : ''));\r\n\t\t\tif (v == null) return;\r\n\r\n\t\t\tvar min = (b && b.min != null) ? Number(b.min) : null;\r\n\t\t\tvar max = (b && b.max != null) ? Number(b.max) : null;\r\n\t\t\tv = clamp_num(v, isNaN(min) ? null : min, isNaN(max) ? null : max);\r\n\r\n\t\t\tif (parts.num)   parts.num.value   = String(v);\r\n\t\t\tif (parts.range) parts.range.value = String(v);\r\n\r\n\t\t\tthis._write_combined(parts, String(v), unit, /*emit*/ false);\r\n\t\t}\r\n\r\n\t\t_write_combined(parts, num, unit, emit) {\r\n\t\t\tif (!parts) return;\r\n\r\n\t\t\tvar combined = build_combined(num, unit);\r\n\r\n\t\t\tif (parts.writer) {\r\n\t\t\t\t// Avoid recursion: mark as internal write.\r\n\t\t\t\tparts.writer.__wpbc_slider_len_internal = true;\r\n\t\t\t\tparts.writer.value = combined;\r\n\t\t\t\tif (emit) emit_input(parts.writer);\r\n\t\t\t\tparts.writer.__wpbc_slider_len_internal = false;\r\n\t\t\t} else if (parts.num) {\r\n\t\t\t\t// If writer is missing, at least notify via number input.\r\n\t\t\t\tif (emit) emit_input(parts.num);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t_sync_group_from_writer(group) {\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts || !parts.writer) return;\r\n\r\n\t\t\tvar raw = String(parts.writer.value || '').trim();\r\n\t\t\tif (!raw) return;\r\n\r\n\t\t\tvar du = this._get_default_unit(group);\r\n\t\t\tvar p  = parse_len_combined(raw, du);\r\n\r\n\t\t\tif (parts.unit)  parts.unit.value  = p.unit;\r\n\t\t\tif (parts.num)   parts.num.value   = p.num;\r\n\t\t\tif (parts.range) parts.range.value = p.num;\r\n\t\t}\r\n\r\n\t\t_on_input(ev) {\r\n\t\t\tvar t = ev.target;\r\n\t\t\tif (!t) return;\r\n\r\n\t\t\tvar group = this._find_group(t);\r\n\t\t\tif (!group) return;\r\n\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts) return;\r\n\r\n\t\t\t// Writer changed externally -> update UI.\r\n\t\t\tif (parts.writer && t === parts.writer) {\r\n\t\t\t\tif (t.__wpbc_slider_len_internal) return;\r\n\t\t\t\tthis._sync_group_from_writer(group);\r\n\t\t\t\tthis._apply_bounds_for_current_unit(group);\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Slider moved -> update number + writer.\r\n\t\t\tif (t.matches && t.matches(this.opts.range_selector)) {\r\n\t\t\t\tif (parts.num) parts.num.value = t.value;\r\n\r\n\t\t\t\tvar unit = (parts.unit && parts.unit.value) ? parts.unit.value : this._get_default_unit(group);\r\n\t\t\t\tthis._write_combined(parts, t.value, unit, /*emit*/ true);\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Number typed -> update slider + writer (clamp if slider has bounds).\r\n\t\t\tif (t.matches && t.matches(this.opts.value_selector)) {\r\n\t\t\t\tvar v = parse_float(t.value);\r\n\r\n\t\t\t\tif (v != null && parts.range) {\r\n\t\t\t\t\tvar rmin = Number(parts.range.min);\r\n\t\t\t\t\tvar rmax = Number(parts.range.max);\r\n\t\t\t\t\tv = clamp_num(v, isNaN(rmin) ? null : rmin, isNaN(rmax) ? null : rmax);\r\n\r\n\t\t\t\t\tparts.range.value = String(v);\r\n\t\t\t\t\tif (String(v) !== t.value) t.value = String(v);\r\n\t\t\t\t}\r\n\r\n\t\t\t\tvar unit2 = (parts.unit && parts.unit.value) ? parts.unit.value : this._get_default_unit(group);\r\n\t\t\t\tthis._write_combined(parts, t.value, unit2, /*emit*/ true);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t_on_change(ev) {\r\n\t\t\tvar t = ev.target;\r\n\t\t\tif (!t) return;\r\n\r\n\t\t\tvar group = this._find_group(t);\r\n\t\t\tif (!group) return;\r\n\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts) return;\r\n\r\n\t\t\t// Unit changed -> update bounds + writer.\r\n\t\t\tif (t.matches && t.matches(this.opts.unit_selector)) {\r\n\t\t\t\tthis._apply_bounds_for_current_unit(group);\r\n\r\n\t\t\t\tvar num  = parts.num ? parts.num.value : (parts.range ? parts.range.value : '');\r\n\t\t\t\tvar unit = t.value || this._get_default_unit(group);\r\n\t\t\t\tthis._write_combined(parts, num, unit, /*emit*/ true);\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\t// -------------------------------------------------------------------------------------------------\r\n\t// Auto-init\r\n\t// -------------------------------------------------------------------------------------------------\r\n\tfunction wpbc_slider_len_groups__auto_init() {\r\n\t\tvar ROOT  = '.wpbc_slider_len_groups';\r\n\t\tvar nodes = Array.prototype.slice.call(d.querySelectorAll(ROOT))\r\n\t\t\t.filter(function (n) { return !n.parentElement || !n.parentElement.closest(ROOT); });\r\n\r\n\t\t// If no explicit containers, install a single document-root instance.\r\n\t\tif (!nodes.length) {\r\n\t\t\tif (!d.__wpbc_slider_len_groups_global_instance) {\r\n\t\t\t\td.__wpbc_slider_len_groups_global_instance = new WPBC_Slider_Len_Groups(d).init();\r\n\t\t\t}\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tnodes.forEach(function (node) {\r\n\t\t\tif (node.__wpbc_slider_len_groups_instance) return;\r\n\t\t\tnode.__wpbc_slider_len_groups_instance = new WPBC_Slider_Len_Groups(node).init();\r\n\t\t});\r\n\t}\r\n\r\n\t// Export globals (manual control if needed).\r\n\tw.WPBC_Slider_Len_Groups   = WPBC_Slider_Len_Groups;\r\n\tw.WPBC_Slider_Len_AutoInit = wpbc_slider_len_groups__auto_init;\r\n\r\n\t// DOM-ready auto init.\r\n\tif (d.readyState === 'loading') {\r\n\t\td.addEventListener('DOMContentLoaded', wpbc_slider_len_groups__auto_init, { once: true });\r\n\t} else {\r\n\t\twpbc_slider_len_groups__auto_init();\r\n\t}\r\n\r\n})(window, document);\r\n","/* globals window, document */\r\n/**\r\n * WPBC Slider Range Groups\r\n *\r\n * Universal, dependency-free controller that keeps a \"range + number\" pair in sync:\r\n *  - number input  (data-wpbc_slider_range_value)\r\n *  - range slider  (data-wpbc_slider_range_range)\r\n *  - writer input  (data-wpbc_slider_range_writer) [optional]\r\n *\r\n * If writer exists: number/slider update writer and emit 'input' on writer (bubbles).\r\n * If writer is missing: emits 'input' on the number input.\r\n * If writer changes externally: updates number/slider.\r\n *\r\n * Markup expectations (minimal):\r\n *  <div class=\"wpbc_slider_range_group\">\r\n *    <input type=\"number\" data-wpbc_slider_range_value>\r\n *    <input type=\"range\"  data-wpbc_slider_range_range>\r\n *    <!-- optional -->\r\n *    <input type=\"text\" data-wpbc_slider_range_writer style=\"display:none;\">\r\n *  </div>\r\n *\r\n * Performance notes:\r\n * - MutationObserver is DISABLED by default.\r\n * - If your UI re-renders and inserts new groups dynamically, call:\r\n *     WPBC_Slider_Range_AutoInit(); OR instance.refresh();\r\n *   Or enable observer via: new WPBC_Slider_Range_Groups(root, { enable_observer:true }).init();\r\n *\r\n * Public API (instance methods):\r\n *  - init(), destroy(), refresh()\r\n *\r\n * @version 2026-01-25\r\n * @since   2026-01-25\r\n * @file    ../includes/__js/admin/slider_groups/wpbc_range_groups.js\r\n */\r\n(function (w, d) {\r\n\t'use strict';\r\n\r\n\t// -------------------------------------------------------------------------------------------------\r\n\t// Helpers\r\n\t// -------------------------------------------------------------------------------------------------\r\n\tfunction clamp_num(v, min, max) {\r\n\t\tif (typeof min === 'number' && !isNaN(min)) v = Math.max(min, v);\r\n\t\tif (typeof max === 'number' && !isNaN(max)) v = Math.min(max, v);\r\n\t\treturn v;\r\n\t}\r\n\r\n\tfunction parse_float(v) {\r\n\t\tvar n = parseFloat(v);\r\n\t\treturn isNaN(n) ? null : n;\r\n\t}\r\n\r\n\tfunction emit_input(el) {\r\n\t\tif (!el) return;\r\n\t\tel.dispatchEvent(new Event('input', { bubbles: true }));\r\n\t}\r\n\r\n\t// -------------------------------------------------------------------------------------------------\r\n\t// Controller\r\n\t// -------------------------------------------------------------------------------------------------\r\n\tclass WPBC_Slider_Range_Groups {\r\n\r\n\t\t/**\r\n\t\t * @param {HTMLElement|string} root_el Container (or selector). If omitted, uses document.\r\n\t\t * @param {Object} [opts={}]\r\n\t\t */\r\n\t\tconstructor(root_el, opts) {\r\n\t\t\tthis.root = root_el\r\n\t\t\t\t? ((typeof root_el === 'string') ? d.querySelector(root_el) : root_el)\r\n\t\t\t\t: d;\r\n\r\n\t\t\tthis.opts = Object.assign({\r\n\t\t\t\t// Strict selectors (NO backward compatibility).\r\n\t\t\t\tgroup_selector  : '.wpbc_slider_range_group',\r\n\t\t\t\tvalue_selector  : '[data-wpbc_slider_range_value]',\r\n\t\t\t\trange_selector  : '[data-wpbc_slider_range_range]',\r\n\t\t\t\twriter_selector : '[data-wpbc_slider_range_writer]',\r\n\r\n\t\t\t\t// Disabled by default for performance.\r\n\t\t\t\tenable_observer     : false,\r\n\t\t\t\tobserver_debounce_ms: 150\r\n\t\t\t}, opts || {});\r\n\r\n\t\t\tthis._on_input  = this._on_input.bind(this);\r\n\t\t\tthis._on_change = this._on_change.bind(this);\r\n\r\n\t\t\tthis._observer    = null;\r\n\t\t\tthis._refresh_tmr = null;\r\n\t\t}\r\n\r\n\t\tinit() {\r\n\t\t\tif (!this.root) return this;\r\n\r\n\t\t\tthis.root.addEventListener('input',  this._on_input,  true);\r\n\t\t\tthis.root.addEventListener('change', this._on_change, true);\r\n\r\n\t\t\tif (this.opts.enable_observer && w.MutationObserver) {\r\n\t\t\t\tthis._observer = new MutationObserver(() => { this._debounced_refresh(); });\r\n\t\t\t\tthis._observer.observe(this.root === d ? d.documentElement : this.root, { childList: true, subtree: true });\r\n\t\t\t}\r\n\r\n\t\t\tthis.refresh();\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\tdestroy() {\r\n\t\t\tif (!this.root) return;\r\n\r\n\t\t\tthis.root.removeEventListener('input',  this._on_input,  true);\r\n\t\t\tthis.root.removeEventListener('change', this._on_change, true);\r\n\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\r\n\t\t\tif (this._refresh_tmr) {\r\n\t\t\t\tclearTimeout(this._refresh_tmr);\r\n\t\t\t\tthis._refresh_tmr = null;\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\trefresh() {\r\n\t\t\tif (!this.root) return;\r\n\r\n\t\t\tvar scope  = (this.root === d ? d : this.root);\r\n\t\t\tvar groups = Array.prototype.slice.call(scope.querySelectorAll(this.opts.group_selector));\r\n\r\n\t\t\tfor (var i = 0; i < groups.length; i++) {\r\n\t\t\t\tthis._sync_from_writer(groups[i]);\r\n\t\t\t\tthis._clamp_to_range(groups[i]);\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\t\t_debounced_refresh() {\r\n\t\t\tif (this._refresh_tmr) clearTimeout(this._refresh_tmr);\r\n\t\t\tthis._refresh_tmr = setTimeout(() => {\r\n\t\t\t\tthis._refresh_tmr = null;\r\n\t\t\t\tthis.refresh();\r\n\t\t\t}, Number(this.opts.observer_debounce_ms) || 0);\r\n\t\t}\r\n\r\n\t\t_find_group(el) {\r\n\t\t\treturn (el && el.closest) ? el.closest(this.opts.group_selector) : null;\r\n\t\t}\r\n\r\n\t\t_get_parts(group) {\r\n\t\t\tif (!group) return null;\r\n\t\t\treturn {\r\n\t\t\t\tgroup : group,\r\n\t\t\t\tnum   : group.querySelector(this.opts.value_selector),\r\n\t\t\t\trange : group.querySelector(this.opts.range_selector),\r\n\t\t\t\twriter: group.querySelector(this.opts.writer_selector)\r\n\t\t\t};\r\n\t\t}\r\n\r\n\t\t_write(parts, value, emit) {\r\n\t\t\tif (!parts) return;\r\n\r\n\t\t\tif (parts.writer) {\r\n\t\t\t\tparts.writer.__wpbc_slider_range_internal = true;\r\n\t\t\t\tparts.writer.value = String(value);\r\n\t\t\t\tif (emit) emit_input(parts.writer);\r\n\t\t\t\tparts.writer.__wpbc_slider_range_internal = false;\r\n\t\t\t} else if (parts.num) {\r\n\t\t\t\t// If writer is missing, at least notify via number input.\r\n\t\t\t\tif (emit) emit_input(parts.num);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t_sync_from_writer(group) {\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts || !parts.writer) return;\r\n\r\n\t\t\tvar raw = String(parts.writer.value || '').trim();\r\n\t\t\tif (!raw) return;\r\n\r\n\t\t\tif (parts.num)   parts.num.value   = raw;\r\n\t\t\tif (parts.range) parts.range.value = raw;\r\n\t\t}\r\n\r\n\t\t_clamp_to_range(group) {\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts || !parts.range || !parts.num) return;\r\n\r\n\t\t\tvar v = parse_float(parts.num.value);\r\n\t\t\tif (v == null) return;\r\n\r\n\t\t\tvar min = Number(parts.range.min);\r\n\t\t\tvar max = Number(parts.range.max);\r\n\t\t\tvar vv  = clamp_num(v, isNaN(min) ? null : min, isNaN(max) ? null : max);\r\n\r\n\t\t\tif (String(vv) !== parts.num.value) parts.num.value = String(vv);\r\n\t\t\tparts.range.value = String(vv);\r\n\t\t}\r\n\r\n\t\t_on_input(ev) {\r\n\t\t\tvar t = ev.target;\r\n\t\t\tif (!t) return;\r\n\r\n\t\t\tvar group = this._find_group(t);\r\n\t\t\tif (!group) return;\r\n\r\n\t\t\tvar parts = this._get_parts(group);\r\n\t\t\tif (!parts) return;\r\n\r\n\t\t\t// Writer changed externally -> update UI.\r\n\t\t\tif (parts.writer && t === parts.writer) {\r\n\t\t\t\tif (t.__wpbc_slider_range_internal) return;\r\n\t\t\t\tthis._sync_from_writer(group);\r\n\t\t\t\tthis._clamp_to_range(group);\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Range moved -> update number + writer.\r\n\t\t\tif (t.matches && t.matches(this.opts.range_selector)) {\r\n\t\t\t\tif (parts.num) parts.num.value = t.value;\r\n\t\t\t\tthis._write(parts, t.value, /*emit*/ true);\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Number typed -> update range + writer (clamp by slider bounds).\r\n\t\t\tif (t.matches && t.matches(this.opts.value_selector)) {\r\n\t\t\t\tif (parts.range) {\r\n\t\t\t\t\tvar v = parse_float(t.value);\r\n\t\t\t\t\tif (v != null) {\r\n\t\t\t\t\t\tvar min = Number(parts.range.min);\r\n\t\t\t\t\t\tvar max = Number(parts.range.max);\r\n\t\t\t\t\t\tv = clamp_num(v, isNaN(min) ? null : min, isNaN(max) ? null : max);\r\n\r\n\t\t\t\t\t\tparts.range.value = String(v);\r\n\t\t\t\t\t\tif (String(v) !== t.value) t.value = String(v);\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t\tthis._write(parts, t.value, /*emit*/ true);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t_on_change(ev) {\r\n\t\t\t// No special \"change\" handling needed currently; kept for symmetry/future.\r\n\t\t}\r\n\t}\r\n\r\n\t// -------------------------------------------------------------------------------------------------\r\n\t// Auto-init\r\n\t// -------------------------------------------------------------------------------------------------\r\n\tfunction wpbc_slider_range_groups__auto_init() {\r\n\t\tvar ROOT  = '.wpbc_slider_range_groups';\r\n\t\tvar nodes = Array.prototype.slice.call(d.querySelectorAll(ROOT))\r\n\t\t\t.filter(function (n) { return !n.parentElement || !n.parentElement.closest(ROOT); });\r\n\r\n\t\tif (!nodes.length) {\r\n\t\t\tif (!d.__wpbc_slider_range_groups_global_instance) {\r\n\t\t\t\td.__wpbc_slider_range_groups_global_instance = new WPBC_Slider_Range_Groups(d).init();\r\n\t\t\t}\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tnodes.forEach(function (node) {\r\n\t\t\tif (node.__wpbc_slider_range_groups_instance) return;\r\n\t\t\tnode.__wpbc_slider_range_groups_instance = new WPBC_Slider_Range_Groups(node).init();\r\n\t\t});\r\n\t}\r\n\r\n\t// Export globals.\r\n\tw.WPBC_Slider_Range_Groups   = WPBC_Slider_Range_Groups;\r\n\tw.WPBC_Slider_Range_AutoInit = wpbc_slider_range_groups__auto_init;\r\n\r\n\tif (d.readyState === 'loading') {\r\n\t\td.addEventListener('DOMContentLoaded', wpbc_slider_range_groups__auto_init, { once: true });\r\n\t} else {\r\n\t\twpbc_slider_range_groups__auto_init();\r\n\t}\r\n\r\n})(window, document);\r\n","/**\r\n * Booking Calendar — Generic UI Tabs Utility (JS)\r\n *\r\n * Purpose: Lightweight, dependency-free tabs controller for any small tab group in admin UIs.\r\n * - Auto-initializes groups marked with data-wpbc-tabs.\r\n * - Assigns ARIA roles and toggles aria-selected/aria-hidden/tabindex.\r\n * - Supports keyboard navigation (Left/Right/Home/End).\r\n * - Public API: window.wpbc_ui_tabs.{init_on, init_group, set_active}\r\n * - Emits 'wpbc:tabs:change' on the group root when the active tab changes.\r\n *\r\n * Markup contract:\r\n * - Root:   [data-wpbc-tabs]\r\n * - Tabs:   [data-wpbc-tab-key=\"K\"]\r\n * - Panels: [data-wpbc-tab-panel=\"K\"]\r\n *\r\n * @package   Booking Calendar\r\n * @subpackage Admin\\UI\r\n * @since     11.0.0\r\n * @version   1.0.0\r\n * @see       /includes/__js/admin/ui_tabs/ui_tabs.js\r\n *\r\n *\r\n * How it works:\r\n * - Root node must have [data-wpbc-tabs] attribute (any value).\r\n * - Tab buttons must carry [data-wpbc-tab-key=\"...\"] (unique per group).\r\n * - Panels must carry [data-wpbc-tab-panel=\"...\"] with matching keys.\r\n * - Adds WAI-ARIA roles and aria-selected/hidden wiring.\r\n *\r\n * <div data-wpbc-tabs=\"column-styles\" data-wpbc-tab-active=\"1\"    class=\"wpbc_ui_tabs_root\" >\r\n *    <!-- Top Tabs -->\r\n *    <div data-wpbc-tablist=\"\" role=\"tablist\"                    class=\" wpbc_ui_el__horis_top_bar__wrapper\" >\r\n *        <div class=\"wpbc_ui_el__horis_top_bar__content\">\r\n *            <h2 class=\"wpbc_ui_el__horis_nav_label\">Column:</h2>\r\n *\r\n *            <div class=\"wpbc_ui_el__horis_nav_item wpbc_ui_el__horis_nav_item__1\">\r\n *                <a\r\n *                    data-wpbc-tab-key=\"1\"\r\n *                    aria-selected=\"true\" role=\"tab\" tabindex=\"0\" aria-controls=\"wpbc_tab_panel_col_1\"\r\n *\r\n *                        href=\"javascript:void(0);\"\r\n *                        class=\"wpbc_ui_el__horis_nav_item__a wpbc_ui_el__horis_nav_item__single\"\r\n *                        id=\"wpbc_tab_col_1\"\r\n *                        title=\"Column 1\"\r\n *                ><span class=\"wpbc_ui_el__horis_nav_title\">Title 1</span></a>\r\n *            </div>\r\n *            ...\r\n *        </div>\r\n *    </div>\r\n *    <!-- Tabs Content -->\r\n *    <div class=\"wpbc_tab__panel group__fields\" data-wpbc-tab-panel=\"1\" id=\"wpbc_tab_panel_col_1\" role=\"tabpanel\" aria-labelledby=\"wpbc_tab_col_1\">\r\n *        ...\r\n *    </div>\r\n *    ...\r\n * </div>\r\n *\r\n * Public API:\r\n *   - wpbc_ui_tabs.init_on(root_or_selector)   // find and init groups within a container\r\n *   - wpbc_ui_tabs.init_group(root_el)         // init a single group root\r\n *   - wpbc_ui_tabs.set_active(root_el, key)    // programmatically change active tab\r\n *\r\n * Events:\r\n *   - Dispatches CustomEvent 'wpbc:tabs:change' on root when tab changes:\r\n *       detail: { active_key: '2', prev_key: '1' }\r\n *\r\n * Switch a local (generic) tabs group to tab 3:     var group = document.querySelector('[data-wpbc-tabs=\"column-styles\"]'); if ( group ) { wpbc_ui_tabs.set_active(group, '3'); }\r\n */\r\n(function ( w ) {\r\n\t'use strict';\r\n\r\n\tif ( w.wpbc_ui_tabs ) {\r\n\t\treturn;\r\n\t}\r\n\r\n\t/**\r\n\t * Internal: toggle active state.\r\n\t *\r\n\t * @param {HTMLElement} root_el\r\n\t * @param {string}      key\r\n\t * @param {boolean}     should_emit\r\n\t */\r\n\tfunction set_active_internal( root_el, key, should_emit ) {\r\n\t\tvar tab_btns = root_el.querySelectorAll( '[data-wpbc-tab-key]' );\r\n\t\tvar panels   = root_el.querySelectorAll( '[data-wpbc-tab-panel]' );\r\n\r\n\t\tvar prev_key = root_el.getAttribute( 'data-wpbc-tab-active' ) || null;\r\n\t\tif ( String( prev_key ) === String( key ) ) {\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// Buttons: aria + class\r\n\t\tfor ( var i = 0; i < tab_btns.length; i++ ) {\r\n\t\t\tvar btn   = tab_btns[i];\r\n\t\t\tvar b_key = btn.getAttribute( 'data-wpbc-tab-key' );\r\n\t\t\tvar is_on = String( b_key ) === String( key );\r\n\r\n\t\t\tbtn.setAttribute( 'role', 'tab' );\r\n\t\t\tbtn.setAttribute( 'aria-selected', is_on ? 'true' : 'false' );\r\n\t\t\tbtn.setAttribute( 'tabindex', is_on ? '0' : '-1' );\r\n\r\n\t\t\tif ( is_on ) {\r\n\t\t\t\tbtn.classList.add( 'active' );\r\n\t\t\t} else {\r\n\t\t\t\tbtn.classList.remove( 'active' );\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t// Panels: aria + visibility\r\n\t\tfor ( var j = 0; j < panels.length; j++ ) {\r\n\t\t\tvar pn   = panels[j];\r\n\t\t\tvar pkey = pn.getAttribute( 'data-wpbc-tab-panel' );\r\n\t\t\tvar show = String( pkey ) === String( key );\r\n\r\n\t\t\tpn.setAttribute( 'role', 'tabpanel' );\r\n\t\t\tpn.setAttribute( 'aria-hidden', show ? 'false' : 'true' );\r\n\t\t\tif ( show ) {\r\n\t\t\t\tpn.removeAttribute( 'hidden' );\r\n\t\t\t} else {\r\n\t\t\t\tpn.setAttribute( 'hidden', '' );\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\troot_el.setAttribute( 'data-wpbc-tab-active', String( key ) );\r\n\r\n\t\tif ( should_emit ) {\r\n\t\t\ttry {\r\n\t\t\t\tvar ev = new w.CustomEvent( 'wpbc:tabs:change', {\r\n\t\t\t\t\tbubbles : true,\r\n\t\t\t\t\tdetail  : { active_key : String( key ), prev_key : prev_key }\r\n\t\t\t\t} );\r\n\t\t\t\troot_el.dispatchEvent( ev );\r\n\t\t\t} catch ( _e ) {}\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Internal: get ordered keys from buttons.\r\n\t *\r\n\t * @param {HTMLElement} root_el\r\n\t * @returns {string[]}\r\n\t */\r\n\tfunction get_keys( root_el ) {\r\n\t\tvar list = [];\r\n\t\tvar btns = root_el.querySelectorAll( '[data-wpbc-tab-key]' );\r\n\t\tfor ( var i = 0; i < btns.length; i++ ) {\r\n\t\t\tvar k = btns[i].getAttribute( 'data-wpbc-tab-key' );\r\n\t\t\tif ( k != null && k !== '' ) {\r\n\t\t\t\tlist.push( String( k ) );\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn list;\r\n\t}\r\n\r\n\t/**\r\n\t * Internal: move focus between tabs using keyboard.\r\n\t *\r\n\t * @param {HTMLElement} root_el\r\n\t * @param {number}      dir  +1 (next) / -1 (prev)\r\n\t */\r\n\tfunction focus_relative( root_el, dir ) {\r\n\t\tvar keys    = get_keys( root_el );\r\n\t\tvar current = root_el.getAttribute( 'data-wpbc-tab-active' ) || keys[0] || null;\r\n\t\tvar idx     = Math.max( 0, keys.indexOf( String( current ) ) );\r\n\t\tvar next    = keys[ ( idx + ( dir > 0 ? 1 : keys.length - 1 ) ) % keys.length ];\r\n\r\n\t\tvar next_btn = root_el.querySelector( '[data-wpbc-tab-key=\"' + next + '\"]' );\r\n\t\tif ( next_btn ) {\r\n\t\t\tnext_btn.focus();\r\n\t\t\tset_active_internal( root_el, next, true );\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Initialize a single tabs group root.\r\n\t *\r\n\t * @param {HTMLElement} root_el\r\n\t */\r\n\tfunction init_group( root_el ) {\r\n\t\tif ( ! root_el || root_el.__wpbc_tabs_inited ) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\troot_el.__wpbc_tabs_inited = true;\r\n\r\n\t\t// Roles\r\n\t\tvar tablist = root_el.querySelector( '[data-wpbc-tablist]' ) || root_el;\r\n\t\ttablist.setAttribute( 'role', 'tablist' );\r\n\r\n\t\t// Default active: from attribute or first button\r\n\t\tvar keys = get_keys( root_el );\r\n\t\tvar def  = root_el.getAttribute( 'data-wpbc-tab-active' ) || ( keys[0] || '1' );\r\n\t\tset_active_internal( root_el, def, false );\r\n\r\n\t\t// Clicks\r\n\t\troot_el.addEventListener( 'click', function ( e ) {\r\n\t\t\tvar btn = e.target.closest ? e.target.closest( '[data-wpbc-tab-key]' ) : null;\r\n\t\t\tif ( ! btn || ! root_el.contains( btn ) ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\te.preventDefault();\r\n\t\t\tvar key = btn.getAttribute( 'data-wpbc-tab-key' );\r\n\t\t\tif ( key != null ) {\r\n\t\t\t\tset_active_internal( root_el, key, true );\r\n\t\t\t}\r\n\t\t}, true );\r\n\r\n\t\t// Keyboard (Left/Right/Home/End)\r\n\t\troot_el.addEventListener( 'keydown', function ( e ) {\r\n\t\t\tvar tgt = e.target;\r\n\t\t\tif ( ! tgt || ! tgt.hasAttribute || ! tgt.hasAttribute( 'data-wpbc-tab-key' ) ) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tswitch ( e.key ) {\r\n\t\t\tcase 'ArrowLeft':\r\n\t\t\t\te.preventDefault(); focus_relative( root_el, -1 ); break;\r\n\t\t\tcase 'ArrowRight':\r\n\t\t\t\te.preventDefault(); focus_relative( root_el, +1 ); break;\r\n\t\t\tcase 'Home':\r\n\t\t\t\te.preventDefault(); set_active_internal( root_el, ( get_keys( root_el )[0] || '1' ), true ); break;\r\n\t\t\tcase 'End':\r\n\t\t\t\te.preventDefault(); var ks = get_keys( root_el ); set_active_internal( root_el, ( ks[ ks.length - 1 ] || '1' ), true ); break;\r\n\t\t\t}\r\n\t\t}, true );\r\n\t}\r\n\r\n\t/**\r\n\t * Initialize all groups within a container (or document).\r\n\t *\r\n\t * @param {HTMLElement|string|null} container\r\n\t */\r\n\tfunction init_on( container ) {\r\n\t\tvar ctx = container ? ( typeof container === 'string' ? document.querySelector( container ) : container ) : document;\r\n\t\tif ( ! ctx ) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tvar groups = ctx.querySelectorAll( '[data-wpbc-tabs]' );\r\n\t\tfor ( var i = 0; i < groups.length; i++ ) {\r\n\t\t\tinit_group( groups[i] );\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Programmatically set active tab by key.\r\n\t *\r\n\t * @param {HTMLElement} root_el\r\n\t * @param {string|number} key\r\n\t */\r\n\tfunction set_active( root_el, key ) {\r\n\t\tif ( root_el && root_el.hasAttribute && root_el.hasAttribute( 'data-wpbc-tabs' ) ) {\r\n\t\t\tset_active_internal( root_el, String( key ), true );\r\n\t\t}\r\n\t}\r\n\r\n\t// Public API (snake_case)\r\n\tw.wpbc_ui_tabs = {\r\n\t\tinit_on    : init_on,\r\n\t\tinit_group : init_group,\r\n\t\tset_active : set_active\r\n\t};\r\n\r\n\t// Auto-init on DOM ready\r\n\tif ( document.readyState === 'loading' ) {\r\n\t\tdocument.addEventListener( 'DOMContentLoaded', function () { init_on( document ); } );\r\n\t} else {\r\n\t\tinit_on( document );\r\n\t}\r\n\r\n})( window );\r\n"]}