Current File : /home/kelaby89/sergio-cuchi.tattoo/wp-content/plugins/so-widgets-bundle/base/js/admin.js |
/* globals wp, jQuery, _, soWidgets, confirm, tinymce, sowbForms */
var sowbForms = window.sowbForms || {};
(function ($) {
const isBlockEditor = $( 'body' ).hasClass( 'block-editor-page' );
let fontList = '';
for ( const [ value, label ] of Object.entries( soWidgets.fonts ) ) {
fontList += '<option value="' + value + '">' + label + '</option>';
}
/**
* Set up font fields when the section is hovered.
*
* This function finds all font fields within the section and appends the font list to each font field.
* It also sets the selected font if a font is already selected.
*/
const setupFontFieldsOnHover = () => {
const $fields = $( this ).find( '> .siteorigin-widget-section > .siteorigin-widget-field-font .siteorigin-widget-input' );
$fields.each( function() {
setupFontField( $( this ) );
} );
}
/**
* Set up a single font field.
*
* This function appends the font list to the given font field and sets the selected font if a font is already selected.
*
* @param {jQuery} $fontSelect - The jQuery object representing the font select element.
*/
const setupFontField = ( $fontSelect ) => {
$fontSelect.append( fontList );
// Set selected font.
var selectedFont = $fontSelect.data( 'selected' );
if ( selectedFont ) {
$fontSelect.val( selectedFont );
}
}
$.fn.sowSetupForm = function () {
return $(this).each(function (i, el) {
var $el = $(el),
$mainForm,
formId,
formInitializing = true;
var $body = $( 'body' );
// Skip this if the widget has any fields with an __i__
var $inputs = $el.find('input[name]');
if ($inputs.length && $inputs.attr('name').indexOf('__i__') !== -1) {
return this;
}
let setupFieldBackups = false;
// Skip this if we've already set up the form
if ( $el.is('.siteorigin-widget-form-main') ) {
if ($el.data('sow-form-setup') === true) {
return true;
}
// If we're in the legacy main widgets interface and the form isn't visible and it isn't contained in a
// panels dialog (when using the Layout Builder widget), don't worry about setting it up.
if (
$body.hasClass( 'widgets-php' ) &&
! isBlockEditor &&
! $el.is( ':visible' ) &&
$el.closest( '.panel-dialog' ).length === 0
) {
return true;
}
// Listen for a state change event if this is the main form wrapper
$el.on('sowstatechange', function (e, incomingGroup, incomingState) {
// Find all wrappers that have state handlers on them
$el.find('[data-state-handler]').each(function () {
var $$ = $(this);
// Create a copy of the current state handlers. Add in initial handlers if the form is initializing.
var handler = $.extend({}, $$.data('state-handler'), formInitializing ? $$.data('state-handler-initial') : {});
if (Object.keys(handler).length === 0) {
return true;
}
// We need to figure out what the incoming state is
var handlerStateParts, handlerState, thisHandler, $$f, runHandler, handlerStateNames;
// Indicates if the handler has run
var handlerRun = {};
var repeaterIndex = sowbForms.getContainerFieldId( $$, 'repeater', '.siteorigin-widget-field-repeater-item' );
if (repeaterIndex !== false) {
var repeaterHandler = {};
for ( var rptrState in handler) {
repeaterHandler[rptrState.replace('{$repeater}', repeaterIndex)] = handler[rptrState];
}
handler = repeaterHandler;
}
var widgetFieldId = sowbForms.getContainerFieldId( $$, 'widget', '.siteorigin-widget-widget' );
if ( widgetFieldId !== false ) {
var widgetFieldHandler = {};
for ( var wdgFldState in handler) {
var stMatches = wdgFldState.match( /_else\[(.*)\]|(.*)\[(.*)\]/ );
var st = '';
if ( stMatches && stMatches.length && stMatches[1] === undefined ) {
st = stMatches[ 2 ] + '_' + widgetFieldId + '[' + stMatches[ 3 ] + ']';
} else {
st = '_else[' + stMatches[ 1 ] + '_' + widgetFieldId + ']';
}
widgetFieldHandler[st] = handler[wdgFldState];
}
handler = widgetFieldHandler;
}
// Go through all the handlers
for (var state in handler) {
runHandler = false;
// Parse the handler state parts
handlerStateParts = state.match(/^([a-zA-Z0-9_-]+)(\[([a-zA-Z0-9_\-,]+)\])?(\[\])?$/);
if (handlerStateParts === null) {
// Skip this if there's a problem with the state parts
continue;
}
handlerState = {
'group': 'default',
'name': '',
'multi': false
};
// Assign the handlerState attributes based on the parsed state
if (handlerStateParts[2] !== undefined) {
handlerState.group = handlerStateParts[1];
handlerState.name = handlerStateParts[3];
}
else {
handlerState.name = handlerStateParts[0];
}
handlerState.multi = (handlerStateParts[4] !== undefined);
if (handlerState.group === '_else') {
// This is the special case of an group else handler
// Always run if no handlers from the current group have been run yet
handlerState.group = handlerState.name;
handlerState.name = '';
// We will run this handler because none have run for it yet
runHandler = ( handlerState.group === incomingGroup && typeof handlerRun[handlerState.group] === 'undefined' );
}
else {
// Evaluate if we're in the current state
handlerStateNames = handlerState.name.split(',').map(function (a) {
return a.trim()
});
for (var i = 0; i < handlerStateNames.length; i++) {
runHandler = (handlerState.group === incomingGroup && handlerStateNames[i] === incomingState);
if (runHandler) break;
}
}
// Run the handler if previous checks have determined we should
if (runHandler) {
thisHandler = handler[state];
// Now we can handle the the handler
if (!handlerState.multi) {
thisHandler = [thisHandler];
}
for (var i = 0; i < thisHandler.length; i++) {
// Choose the item we'll be acting on here
if (typeof thisHandler[i][1] !== 'undefined' && Boolean(thisHandler[i][1])) {
// thisHandler[i][1] is the sub selector
$$f = $$.find(thisHandler[i][1]);
}
else {
$$f = $$;
}
var animated = false;
// Prevent animations from happening on load.
if ( $$f.prop( 'style' ).length ) {
if ( thisHandler[i][0] == 'show' ) {
$$f.fadeIn( 'fast' );
animated = true;
} else if ( thisHandler[i][0] == 'hide' ) {
$$f.fadeOut( 'fast' );
animated = true;
}
}
if ( ! animated ) {
// Call the function on the wrapper we've selected
$$f[ thisHandler[i][0] ].apply( $$f, typeof thisHandler[i][2] !== 'undefined' ? thisHandler[i][2] : [] );
}
if ( $$f.is( '.siteorigin-widget-field:visible' ) ) {
if ( $$f.is( '.siteorigin-widget-field-type-section' ) ) {
var $fields = $$f.find( '> .siteorigin-widget-section > .siteorigin-widget-field' );
$fields.trigger( 'sowsetupformfield' );
} else {
$$f.trigger( 'sowsetupformfield' );
}
}
}
// Store that we've run a handler
handlerRun[handlerState.group] = true;
}
}
});
});
// Lets set up the preview
$el.sowSetupPreview();
$mainForm = $el;
var $teaser = $el.find('.siteorigin-widget-teaser');
$teaser.find( '.dashicons-dismiss' ).on( 'click', function() {
var $$ = $(this);
$.get($$.data('dismiss-url'));
$teaser.slideUp('normal', function () {
$teaser.remove();
});
});
if (
soWidgets.backup.enabled &&
! $el.data( 'backupDisabled' )
) {
setupFieldBackups = true;
}
}
else {
$mainForm = $el.closest('.siteorigin-widget-form-main');
}
formId = $mainForm.find('> .siteorigin-widgets-form-id').val();
// Find any field or sub widget fields.
var $fields = $el.find('> .siteorigin-widget-field');
// Process any sub sections
$fields.find('> .siteorigin-widget-section').sowSetupForm();
var $subwidgetFields = $fields.find('> .siteorigin-widget-widget');
$subwidgetFields.find('> .siteorigin-widget-section').sowSetupForm();
// Process any sub widgets whose fields aren't contained in a section
$subwidgetFields.filter(':not(:has(> .siteorigin-widget-section))').sowSetupForm();
// Store the field names
$fields.find('.siteorigin-widget-input').each(function (i, input) {
if ($(input).data('original-name') === null) {
$(input).data('original-name', $(input).attr('name'));
}
});
// Setup all the repeaters
$fields.find('> .siteorigin-widget-field-repeater').sowSetupRepeater();
// For any repeater items currently in existence.
$el.find('.siteorigin-widget-field-repeater-item').sowSetupRepeaterItems();
// Set up any font fields.
if (
$fields.find( '> .siteorigin-widget-font-selector' ).length &&
fontList
) {
const $fontSelect = $fields.find( '> .siteorigin-widget-font-selector select' );
$fontSelect.each( function() {
const sectionParent = $( this ).closest( '.siteorigin-widget-field-type-section' );
// If the font isn't in a section, set it up immediately.
if ( ! sectionParent.length ) {
setupFontField( $( this ) );
return;
}
// Is the section visible? If so, set it up after 200ms.
if ( sectionParent.find( '> .siteorigin-widget-section-visible' ) ) {
setTimeout( () => {
setupFontField( $( this ) );
}, 200 );
return;
}
// If this section has already had its event setup, skip it.
if ( sectionParent.data( 'font-setup' ) ) {
return;
}
sectionParent.data( 'font-setup', true );
sectionParent.one( 'mouseover', setupFontFieldsOnHover );
} );
}
// Set up any color fields.
$fields.find( '> .siteorigin-widget-input-color' ).each( function() {
var $colorField = $( this );
var colorResult = ''
var alphaImage = '';
if ( $colorField.data( 'alpha-enabled' ) ) {
var handleAlphaDefault = function() {
if ( colorResult == '' ) {
$container = $colorField.parents( '.wp-picker-container' );
$colorResult = $container.find( '.wp-color-result' );
alphaImage = $colorResult.css( 'background-image' );
}
$colorResult.css( 'background-image', $colorField.val() == '' ? 'none' : alphaImage );
}
}
var $colorFieldOptions = {
change: function( event, ui ) {
setTimeout( function() {
if ( $colorField.data( 'alpha-enabled' ) ) {
handleAlphaDefault();
}
$( event.target ).trigger( 'change' );
}, 100 );
}
};
if ( $colorField.data( 'defaultColor' ) ) {
$colorFieldOptions.defaultColor = $colorField.data( 'defaultColor' );
}
if ( $colorField.data( 'palettes' ) ) {
$colorFieldOptions.palettes = $colorField.data( 'palettes' );
}
if ( typeof $.fn.wpColorPicker === 'function' ) {
$colorField.wpColorPicker( $colorFieldOptions );
if ( $colorField.data( 'alpha-enabled' ) ) {
$colorField.on( 'change', handleAlphaDefault ).trigger( 'change' );
}
}
} );
///////////////////////////////////////
// Handle the sections
var expandContainer = function ( e ) {
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
const $this = $( this );
$this.toggleClass( 'siteorigin-widget-section-visible' );
const $section = $this.parent().find( '> .siteorigin-widget-section, > .siteorigin-widget-widget > .siteorigin-widget-section' );
$section.slideToggle( 'fast', function() {
const $thisSection = $( this );
$thisSection.find( '> .siteorigin-widget-field-container-state' ).val( $thisSection.is( ':visible' ) ? 'open' : 'closed' );
if ( $thisSection.is( ':visible' ) ) {
$thisSection.find( '> .siteorigin-widget-field' ).trigger( 'sowsetupformfield' );
}
} );
};
$fields.filter( '.siteorigin-widget-field-type-widget, .siteorigin-widget-field-type-section' ).find( '> label' )
.on( 'click keyup', expandContainer )
.attr( 'tabindex', 0 );
$fields.filter( '.siteorigin-widget-field-type-posts' ).find( '.posts-container-label-wrapper' ).on( 'click keyup', expandContainer );
///////////////////////////////////////
// Handle the slider fields
$fields.filter('.siteorigin-widget-field-type-slider').each(function () {
var $$ = $(this);
var $input = $$.find('input[type="number"]');
var $c = $$.find('.siteorigin-widget-value-slider');
$c.slider({
max: parseFloat($input.attr('max')),
min: parseFloat($input.attr('min')),
step: parseFloat($input.attr('step')),
value: parseFloat($input.val()),
slide: function (event, ui) {
$input.val( parseFloat( ui.value ) );
$input.trigger( 'change' );
$$.find('.siteorigin-widget-slider-value').html(ui.value);
},
});
$input.on( 'change', function( event, data ) {
if ( ! ( data && data.silent ) ) {
$c.slider( 'value', parseFloat( $input.val() ) );
$$.find('.siteorigin-widget-slider-value').html( $input.val() );
}
});
});
///////////////////////////////////////
// Setup the URL fields
$fields.filter( '.siteorigin-widget-field-type-link' ).each( function () {
const $$ = $( this );
const $fieldVal = $$.find( 'input.siteorigin-widget-input' );
const $itemList = $$.find( 'ul.posts' );
const $noResults = $$.find( '.content-no-results' );
let request = null;
const refreshList = () => {
if ( request !== null ) {
request.abort();
}
const $contentSearchInput = $$.find( '.content-text-search' );
const query = $contentSearchInput.val();
const postTypes = $contentSearchInput.data( 'postTypes' );
const ajaxData = {
action: 'so_widgets_search_posts',
query: query,
postTypes: postTypes
};
// If WPML is enabled for this page, include page language for filtering.
if ( typeof icl_this_lang == 'string' ) {
ajaxData.language = icl_this_lang;
}
// Visually prep the field.
$noResults.addClass( 'hidden' );
$itemList.empty();
$itemList.removeClass( 'hidden' )
$itemList.addClass( 'loading' );
$.get(
soWidgets.ajaxurl,
ajaxData,
( results ) => {
// If there aren't any results, show a message.
if ( results.length === 0 ) {
$noResults.removeClass( 'hidden' );
$itemList.addClass( 'hidden' );
$itemList.removeClass( 'loading' );
return;
}
for ( var i = 0; i < results.length; i++ ) {
if (results[ i ].label === '') {
results[ i ].label = ' ';
}
// Add all the post items
$itemList.append(
$( '<li>' )
.addClass( 'post' )
.html( results[ i ].label + '<span>(' + results[ i ].type + ')</span>' )
.data( results[ i ] )
.attr( 'tabindex', 0 )
);
}
$itemList.removeClass( 'loading' );
}
);
};
// Toggle display of the existing content
$$.find( '.select-content-button, .button-close' ).on( 'click', function( e ) {
e.preventDefault();
$( this ).trigger( 'blur' );
var $s = $$.find( '.existing-content-selector' );
$s.toggle();
if ( $s.is( ':visible' ) && $s.find( 'ul.posts li' ).length === 0 ) {
refreshList();
}
});
// Clicking on one of the url items
$$.on( 'click keyup', '.posts li', function( e ) {
e.preventDefault();
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
var $li = $( this );
$fieldVal.val( 'post: ' + $li.data( 'value' ) + ' (' + $li.get(0).childNodes[0].nodeValue + ')' );
$$.trigger( 'change' );
$$.find( '.existing-content-selector' ).toggle();
} );
var interval = null;
$$.find( '.content-text-search' ).on( 'keyup', function() {
if (interval !== null) {
clearTimeout(interval);
}
interval = setTimeout(function () {
refreshList();
}, 500);
});
var linkFieldId = $fieldVal.val().replace( 'post: ', '' );
if ( linkFieldId != '' && isFinite( linkFieldId ) ) {
$.get(
soWidgets.ajaxurl,
{
action: 'so_widgets_links_get_title',
postId: linkFieldId,
},
function( data ) {
$fieldVal.val( $fieldVal.val() + ' (' + data + ')' );
}
);
}
} );
///////////////////////////////////////
// Setup the Builder fields
if (typeof jQuery.fn.soPanelsSetupBuilderWidget !== 'undefined') {
$fields.filter('.siteorigin-widget-field-type-builder').each(function () {
$( this ).find( '> .siteorigin-page-builder-field' ).each( function () {
var $$ = $( this );
$$.soPanelsSetupBuilderWidget( { builderType: $$.data( 'type' ) } );
} );
});
}
///////////////////////////////////////
// Now lets handle the state emitters
var stateEmitterChangeHandler = function () {
var $$ = $(this);
// These emitters can either be an array or a
var emitters = $$.closest('[data-state-emitter]').data('state-emitter');
if (typeof emitters !== 'undefined') {
var handleStateEmitter = function (emitter, currentStates) {
if (typeof sowEmitters[emitter.callback] === 'undefined' || emitter.callback.substr(0, 1) === '_') {
// Skip if the function doesn't exist, or it starts with an underscore (internal functions).
return currentStates;
}
// Skip if this is an unselected radio input.
if ( $$.is( '[type="radio"]' ) && !$$.is( ':checked' ) ) {
return currentStates;
}
// Check if this is inside a repeater
var repeaterIndex = sowbForms.getContainerFieldId( $$, 'repeater', '.siteorigin-widget-field-repeater-item' );
if (repeaterIndex !== false) {
emitter.args = emitter.args.map(function (a) {
return a.replace('{$repeater}', repeaterIndex);
});
}
var widgetFieldId = sowbForms.getContainerFieldId( $$, 'widget', '.siteorigin-widget-widget' );
if ( widgetFieldId !== false && ! emitter.hasOwnProperty( 'widgetFieldId' ) ) {
emitter.widgetFieldId = widgetFieldId;
emitter.args = emitter.args.map(function (arg) {
if ( emitter.callback === 'conditional' ) {
arg = arg.replace( /(.*)(\[.*)/, '$1_' + widgetFieldId + '$2' );
} else {
arg = arg + '_' + widgetFieldId;
}
return arg;
});
}
var val = $$.is('[type="checkbox"]') ? $$.is(':checked') : $$.val();
// Media form fields can have an external field set so we need to check that field slightly differently.
if ( $$.parent().hasClass( 'siteorigin-widget-field-type-media' ) && emitter.callback == 'conditional' ) {
// If we're checking for a value,and the main field is empty,
// fallback to the external field value. This also works in reverse.
if ( ! val ) {
val = $$.hasClass( 'media-fallback-external' ) ? $$.prev().val() : fallbackField = $$.next().val();
}
// Override value if media value is set to 0 to prevent unintentional conditional passing.
if ( val == 0 ) {
val = '';
}
}
// Return an array that has the new states added to the array
return $.extend(
currentStates,
sowEmitters[ emitter.callback ] (
val,
emitter.args,
$$
)
);
};
// Run the states through the state emitters
var states = {'default': ''};
// Go through the array of emitters
if (typeof emitters.length === 'undefined') {
emitters = [emitters];
}
for (var i = 0; i < emitters.length; i++) {
states = handleStateEmitter(emitters[i], states);
}
// Check which states have changed and trigger appropriate sowstatechange
var formStates = $mainForm.data('states');
if (typeof formStates === 'undefined') {
formStates = {'default': ''};
}
for (var k in states) {
if ( typeof formStates[k] === 'undefined' || states[k] !== formStates[k] ) {
// If the state is different from the original formStates, then trigger a state change
formStates[k] = states[k];
$mainForm.trigger('sowstatechange', [k, states[k]]);
}
}
// Store the form states back in the form
$mainForm.data('states', formStates);
}
};
$fields.filter('[data-state-emitter]').each(function () {
var $input = $( this ).find( '.siteorigin-widget-input' );
// Listen for any change events on an emitter field
$input.on('keyup change', stateEmitterChangeHandler);
// Trigger initial state emitter changes
$input.each(function () {
var $$ = $(this);
if ($$.is(':radio')) {
// Only checked radio inputs must have change events
if ($$.is(':checked')) {
stateEmitterChangeHandler.call($$[0]);
}
}
else {
stateEmitterChangeHandler.call($$[0]);
}
});
});
if ( setupFieldBackups ) {
$el.sowFieldBackups( $mainForm );
}
// Give plugins a chance to influence the form
$el.trigger('sowsetupform', $fields).data('sow-form-setup', true);
$fields.trigger('sowsetupformfield');
$el.find('.siteorigin-widget-field-repeater-item').trigger('updateFieldPositions');
if ( $body.hasClass( 'wp-customizer' ) || $body.hasClass( 'widgets-php' ) ) {
// Reinitialize widget fields when they're dragged and dropped.
$el.closest( '.ui-sortable' ).on( 'sortstop', function (event, ui) {
var $fields = ui.item.find( '.siteorigin-widget-form' ).find( '> .siteorigin-widget-field' );
$fields.trigger( 'sowsetupformfield' );
} );
}
/////////////////////////////
// The end of the form setup.
/////////////////////////////
formInitializing = false;
});
};
$.fn.sowSetupPreview = function () {
var $el = $(this);
var previewButton = $el.siblings('.siteorigin-widget-preview');
previewButton.find( '> a' ).on( 'click', function( e ) {
e.preventDefault();
var data = sowbForms.getWidgetFormValues($el);
// Create a new modal window
var modal = $($('#so-widgets-bundle-tpl-preview-dialog').html().trim()).appendTo('body');
modal.find('input[name="data"]').val(JSON.stringify(data));
modal.find('input[name="class"]').val($el.data('class'));
modal.find('iframe').on('load', function () {
$(this).css('visibility', 'visible');
});
modal.find( 'form' ).trigger( 'submit' );
modal.find('.close').on( 'click keyup', function (e) {
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
modal.remove();
});
});
};
$.fn.sowSetupRepeater = function () {
return $(this).each(function (i, el) {
var $el = $(el);
var $items = $el.find('.siteorigin-widget-field-repeater-items');
var name = $el.data('repeater-name');
var maxItems = $el.data( 'max-items' );
$items.on( 'updateFieldPositions', function() {
var $$ = $(this);
var $rptrItems = $$.find('> .siteorigin-widget-field-repeater-item');
// Set the position for the repeater items
$rptrItems.each(function (i, el) {
$(el).find('.siteorigin-widget-input').each(function (j, input) {
var pos = $(input).data('repeater-positions');
if (typeof pos === 'undefined') {
pos = {};
}
pos[name] = i;
$(input).data('repeater-positions', pos);
});
});
// Update the field names for all the input items
$$.find('.siteorigin-widget-input').each( function ( i, input ) {
var $in = $( input );
var pos = $in.data( 'repeater-positions' );
if ( typeof pos !== 'undefined' ) {
var newName = $in.attr( 'data-original-name' );
if ( !newName ) {
$in.attr( 'data-original-name', $in.attr( 'name' ) );
newName = $in.attr( 'name' );
}
if ( !newName ) {
return;
}
if ( pos ) {
for ( var k in pos ) {
newName = newName.replace( '#' + k + '#', pos[ k ] );
}
}
$in.attr( 'name', newName );
}
} );
if ( !$$.data( 'initialSetup' ) ) {
// Setup default checked values, now that we've updated input names.
// Without this radio inputs in repeaters will be rendered as if they all belong to the same group.
$$.find('.siteorigin-widget-input').each(function (i, input) {
var $in = $(input);
$in.prop('checked', $in.prop('defaultChecked'));
});
$$.data('initialSetup', true);
}
//Setup scrolling.
var scrollCount = $el.data('scroll-count') ? parseInt($el.data('scroll-count')) : 0;
if (scrollCount > 0 && $rptrItems.length > scrollCount) {
var itemHeight = $rptrItems.first().outerHeight();
$$.css( 'max-height', itemHeight * scrollCount + 'px' );
$$.css( 'overflow', 'auto' );
}
else {
//TODO: Check whether there was a value before overriding and set it back to that.
$$.css('max-height', '').css('overflow', '');
}
});
$items.sortable({
handle: '.siteorigin-widget-field-repeater-item-top',
items: '> .siteorigin-widget-field-repeater-item',
update: function () {
// Clear `name` attributes for radio inputs. They'll be reassigned on update.
// This prevents some radio inputs values being cleared during the update process.
$items.find( 'input[type="radio"].siteorigin-widget-input' ).attr( 'name', '' );
$items.trigger('updateFieldPositions');
$el.trigger( 'change' );
},
sortstop: function (event, ui) {
if ( ui.item.is( '.siteorigin-widget-field-repeater-item' ) ) {
ui.item.find( '> .siteorigin-widget-field-repeater-item-form' ).each( function () {
var $fields = $( this ).find( '> .siteorigin-widget-field' );
$fields.trigger( 'sowsetupformfield' );
} );
}
else {
var $fields = ui.item.find( '.siteorigin-widget-form' ).find( '> .siteorigin-widget-field' );
$fields.trigger( 'sowsetupformfield' );
}
$el.trigger( 'change' );
}
});
$items.trigger('updateFieldPositions');
var preventNewItems = function() {
$el.addClass( 'sow-max-reached' );
}
$el.find( '> .siteorigin-widget-field-repeater-add' ).disableSelection().on( 'click keyup', function( e ) {
e.preventDefault();
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
if ( isNaN( maxItems ) || $el.find( '.siteorigin-widget-field-repeater-item' ).length + 1 <= maxItems ) {
$el.closest( '.siteorigin-widget-field-repeater' )
.sowAddRepeaterItem()
.find( '> .siteorigin-widget-field-repeater-items' ).slideDown( 'fast', function () {
$( window ).trigger( 'resize' );
} );
if ( isFinite( maxItems ) ) {
if ( $items.find( '.siteorigin-widget-field-repeater-item' ).length == maxItems ) {
preventNewItems();
}
}
} else {
preventNewItems();
}
});
$el.find( '> .siteorigin-widget-field-repeater-top > .siteorigin-widget-field-repeater-expand' ).on( 'click', function( e ) {
e.preventDefault();
$el.closest('.siteorigin-widget-field-repeater').find('> .siteorigin-widget-field-repeateritems-').slideToggle('fast', function () {
$( window ).trigger( 'resize' );
});
});
// Setup Repeater Table Header if necessary.
const itemLabel = $el.data( 'item-label' );
if ( itemLabel !== undefined && 'table' in itemLabel ) {
$el.addClass( 'sow-repeater-has-table' );
let labels = itemLabel.selectorArray.map( item => item.label || '' );
let listItems = labels.map( label => `<li role="listitem">${ limitTextLength( label ) }</li>`).join( '' );
$el.find( '.siteorigin-widget-field-repeater-top' )
.append( `<ul class="sow-repeater-table" role="list" aria-label="${ soWidgets.table.header }">${ listItems }</ul>` )
.append( `<span class="sow-repeater-table-actions">${ soWidgets.table.actions }</span>` );
}
});
};
$.fn.sowAddRepeaterItem = function () {
return $(this).each(function (i, el) {
var $el = $(el);
var $nextIndex = $el.find('> .siteorigin-widget-field-repeater-items').children().length + 1;
// Create an object with the repeater html so we can make some changes to it.
var repeaterObject = $('<div>' + $el.find('> .siteorigin-widget-field-repeater-item-html').html() + '</div>');
repeaterObject.find('.siteorigin-widget-input[data-name]').each(function () {
var $$ = $(this);
// Skip out items that are themselves inside repeater HTML wrappers
if ($$.closest('.siteorigin-widget-field-repeater-item-html').length === 0) {
$$.attr('name', $(this).data('name'));
}
});
// Replace repeater item id placeholders with the index of the repeater item.
var repeaterHtml = '';
repeaterObject.find( '> .siteorigin-widget-field' )
.each( function ( index, element ) {
var html = element.outerHTML;
// Skip child repeaters, so they can setup their own id's when necessary.
if ( ! $( element ).is( '.siteorigin-widget-field-type-repeater' ) ) {
html = html.replace( /_id_/g, $nextIndex );
}
repeaterHtml += html;
} );
var readonly = typeof $el.attr('readonly') !== 'undefined';
var item = $( '<div class="siteorigin-widget-field-repeater-item ui-draggable"></div>' )
.append(
$( '<div class="siteorigin-widget-field-repeater-item-top" tabindex="0" />' )
.append(
$( '<div class="siteorigin-widget-field-expand" tabindex="0" />' )
)
.append(
readonly ? '' : $( '<div class="siteorigin-widget-field-copy" tabindex="0" />' )
)
.append(
readonly ? '' : $( '<div class="siteorigin-widget-field-remove" tabindex="0" />' )
)
.append( $( '<h4></h4>' ).html( $el.data( 'item-name' ) ) )
)
.append(
$( '<div class="siteorigin-widget-field-repeater-item-form"></div>' )
.html(repeaterHtml)
);
// Add the item and refresh
$el.find( '> .siteorigin-widget-field-repeater-items' ).append( item ).sortable( 'refresh' ).trigger( 'updateFieldPositions' );
item.sowSetupRepeaterItems();
item.hide().slideDown( 'fast', function () {
$( window ).trigger( 'resize' );
});
$el.trigger( 'change' );
});
};
$.fn.sowRemoveRepeaterItem = function () {
return $(this).each(function (i, el) {
var $itemsContainer = $(this).closest('.siteorigin-widget-field-repeater-items');
$(this).remove();
$itemsContainer.sortable("refresh").trigger('updateFieldPositions');
$( el ).trigger( 'change' );
});
};
$.fn.checkboxFormField = function() {
const icon = $( this ).is( ':checked' ) ? 'yes' : 'minus';
return `<span class="dashicons dashicons-${ icon }"></span>`;
}
$.fn.iconFormField = function() {
return $( this ).find( '.siteorigin-widget-icon span[data-sow-icon]' ).prop( 'outerHTML' );
}
const limitTextLength = function( text ) {
if ( typeof text === 'undefined' ) {
return '';
}
// Escape the text.
text = $( '<div></div>' ).text( text ).html();
if ( text.length > 80 ) {
return text.substr( 0, 79 ) + '...';
}
return text;
}
const setTextBasedOnType = function( text, type ) {
if (
type === 'iconFormField' ||
type === 'checkboxFormField'
) {
return text;
}
return limitTextLength( text );
}
$.fn.sowSetupRepeaterItems = function () {
return $(this).each(function (i, el) {
var $el = $(el);
if ( typeof $el.data( 'sowrepeater-actions-setup' ) === 'undefined' ) {
var $parentRepeater = $el.closest('.siteorigin-widget-field-repeater');
var itemTop = $el.find('> .siteorigin-widget-field-repeater-item-top');
var itemLabel = $parentRepeater.data('item-label');
var defaultLabel = $el.parents('.siteorigin-widget-field-repeater').data('item-name');
if ( itemLabel && ( itemLabel.hasOwnProperty( 'selector' ) || itemLabel.hasOwnProperty( 'selectorArray' ) ) ) {
var updateLabel = function () {
const isTable = itemLabel !== undefined && 'table' in itemLabel;
var functionName, text, selectorRow;
if ( isTable ) {
var table = [];
}
if ( itemLabel.hasOwnProperty( 'selectorArray' ) ) {
for ( var i = 0 ; i < itemLabel.selectorArray.length ; i++ ) {
selectorRow = itemLabel.selectorArray[ i ];
functionName = ( selectorRow.hasOwnProperty( 'valueMethod' ) && selectorRow.valueMethod ) ? selectorRow.valueMethod : 'val';
let foundText = '';
// Is this a single selector or multiple?
if ( ! isTable || ! selectorRow.selectorArray ) {
foundText = $el.find( selectorRow.selector )[ functionName ]();
} else {
// This item has multiple selectors to check.
// Set foundText to the first valid value found.
for ( var j = 0 ; j < selectorRow.selectorArray.length ; j++ ) {
const selector = selectorRow.selectorArray[ j ];
foundText = $el.find( selector.selector )[ selector.valueMethod ]();
if ( foundText ) {
break;
}
}
}
if ( isTable ) {
// No matter what, we need to push this value for consistent spacing.
table.push( {
value: foundText,
type: selectorRow.valueMethod,
} );
} else if ( foundText ) {
text = text ? `${ text } ${ foundText }` : foundText;
break;
}
}
} else {
functionName = ( itemLabel.hasOwnProperty( 'valueMethod' ) && itemLabel.valueMethod ) ? itemLabel.valueMethod : 'val';
text = $el.find( itemLabel.selector )[ functionName ]();
}
if ( isTable ) {
// Ensure the table is present.
if ( ! itemTop.find( '.sow-repeater-table' ).length ) {
itemTop.find( 'h4' ).after( '<ul class="sow-repeater-table" role="list"></ul>' );
itemTop.find( 'h4' ).remove();
}
let listItems = '';
table.forEach( ( item, index ) => {
text = setTextBasedOnType( item.value, item.type );
listItems += `<li role="listitem">${ text }</li>`;
} );
itemTop.find( '.sow-repeater-table' ).empty().append( listItems );
} else if ( ! isTable && text ) {
text = setTextBasedOnType( text, functionName );
} else {
text = defaultLabel;
// Add item index to label if needed.
if ( itemLabel.increment ) {
// Get the index of the item and avoid the zero-index.
var index = $el.index() + 1;
if ( ! isNaN( index ) ) {
text = itemLabel.increment === 'before' ? `${ index } ${ text }` : `${ text } ${ index }`;
}
}
}
if ( ! text ) {
return;
}
if ( functionName === 'checkboxFormField' ) {
itemTop.find( 'h4' ).html( text );
return;
}
if ( functionName === 'iconFormField' ) {
// There's a chance the default label could show up unexpectedly. Skip it.
if ( text == 'Item' ) {
return;
}
const $item = $( text );
if ( ! $item ) {
return;
}
// If the icon is hidden, it's been removed.
if ( $item.css('display') === 'none' ) {
itemTop.find( 'h4' ).text( defaultLabel );
return;
}
itemTop.find( 'h4' ).html( text );
return;
}
itemTop.find( 'h4' ).text( text );
};
updateLabel();
var eventName = ( itemLabel.hasOwnProperty('updateEvent') && itemLabel.updateEvent ) ? itemLabel.updateEvent : 'change';
$el.on( eventName, updateLabel );
}
itemTop.on( 'click keyup', function( e ) {
if ( e.target.className === 'siteorigin-widget-field-remove' || e.target.className === 'siteorigin-widget-field-copy' ) {
return;
}
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
e.preventDefault();
$( this ).closest( '.siteorigin-widget-field-repeater-item' ).find( '.siteorigin-widget-field-repeater-item-form' ).eq( 0 ).slideToggle( 'fast', function() {
$( window ).trigger( 'resize' );
if ( $ ( this ).is( ':visible' ) ) {
$( this ).trigger( 'slideToggleOpenComplete' );
$( this ).find( '.siteorigin-widget-field-type-section > .siteorigin-widget-section > .siteorigin-widget-field,> .siteorigin-widget-field' )
.each( function (index, element) {
var $field = $( element );
if ( $field.is( ':visible' ) ) {
$field.trigger( 'sowsetupformfield' );
}
} );
} else {
$( this ).trigger( 'slideToggleCloseComplete' );
}
} );
} );
itemTop.find( '.siteorigin-widget-field-remove' ).on( 'click keyup', function( e, params ) {
e.preventDefault();
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
var $s = $( this ).closest( '.siteorigin-widget-field-repeater-items' );
var $item = $( this ).closest( '.siteorigin-widget-field-repeater-item' );
var removeItem = function () {
$item.remove();
$s.sortable( "refresh" ).trigger( 'updateFieldPositions' );
$( window ).trigger( 'resize' );
$parentRepeater.trigger( 'change' );
};
if ( params && params.silent ) {
removeItem();
} else if ( confirm( soWidgets.sure ) ) {
$item.slideUp('fast', removeItem );
// If increment is enabled for this item, trigger label updates.
var itemLabel = $el.closest( '.siteorigin-widget-field-repeater' ).data( 'item-label' );
if ( typeof itemLabel.increment == 'string' ) {
$el.parent().find( '.siteorigin-widget-field-repeater-item' ).trigger( 'change' )
}
// Check if we need to re-enable actions due to no longer being at the maximum number of items.
var $repeater = $( this ).parents('.siteorigin-widget-field-repeater');
if ( $repeater.hasClass( 'sow-max-reached' ) ) {
$repeater.removeClass( 'sow-max-reached' );
}
}
} );
itemTop.find( '.siteorigin-widget-field-copy' ).on( 'click keyup', function( e ) {
e.preventDefault();
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
var $items = $( this ).closest('.siteorigin-widget-field-repeater-items');
var $mainRepeater = $( this ).parents('.siteorigin-widget-field-repeater');
var maxItems = $mainRepeater.data( 'max-items' );
if ( isNaN( maxItems ) || $items.find( '.siteorigin-widget-field-repeater-item' ).length + 1 <= maxItems ) {
var $form = $( this ).closest( '.siteorigin-widget-form-main' );
var $item = $( this ).closest( '.siteorigin-widget-field-repeater-item' );
var $copyItem = $item.clone();
var $nextIndex = $items.children().length;
var newIds = {};
$copyItem.find( '*[name]' ).each(function () {
var $inputElement = $( this );
var id = $inputElement.attr( 'id' );
var nm = $inputElement.attr( 'name' );
// TinyMCE field :/
if ($inputElement.is( 'textarea' ) && $inputElement.parent().is( '.wp-editor-container' ) && typeof tinymce != 'undefined' ) {
$inputElement.parent().empty().append( $inputElement );
$inputElement.css( 'display', '' );
var curEd = tinymce.get( id );
if ( curEd ) {
var contentVal = curEd.getContent();
if ( ! _.isEmpty( contentVal ) ) {
$inputElement.val( contentVal );
} else if ( contentVal.search( '<' ) !== -1 && contentVal.search( '>' ) === -1) {
$textarea.val( contentVal.replace( /</g, '' ) );
}
}
}
// Color field :/
else if ($inputElement.is( '.wp-color-picker' ) ) {
var $wpPickerContainer = $inputElement.closest( '.wp-picker-container' );
var $soWidgetField = $inputElement.closest( '.siteorigin-widget-field' );
$wpPickerContainer.remove();
$soWidgetField.append( $inputElement.remove() );
}
else {
var $originalInput = id ? $item.find( '#' + id ) : $item.find( '[name="' + nm + '"]' );
if ( $originalInput.length && $originalInput.val() != null ) {
$inputElement.val( $originalInput.val() );
}
}
if ( id ) {
var idRegExp;
var idBase;
var newId;
// Radio inputs are slightly different because there are multiple `input` elements for
// a single field, i.e. multiple `inputs` for selecting a single value.
if ( $inputElement.is( '[type="radio"]' ) ) {
// Radio inputs have their position appended to the id.
idBase = id.replace( /-\d+-\d+$/, '' );
var radioIdBase = id.replace( /-\d+$/, '' );
if ( !newIds[ idBase ] ) {
var radioNames = {};
newIds[ idBase ] = $form
// find all inputs containing idBase in their id attribute
.find( '.siteorigin-widget-input[id^=' + idBase + ']' )
// exclude inputs from templates
.not( '[id*=_id_]' )
// reduce to one element per radio input group.
.filter( function( index, element ) {
var eltName = $( element ).attr( 'name' );
if ( radioNames[ eltName] ) {
return false;
} else {
radioNames[ eltName ] = true;
return true;
}
}).length + 1;
}
var newRadioIdBase = idBase + '-' + newIds[ idBase ];
newId = newRadioIdBase + id.match( /-\d+$/ )[0];
$copyItem.find( 'label[for=' + radioIdBase + ']' ).attr( 'for', newRadioIdBase );
} else {
idRegExp = new RegExp( '-\\d+$' );
idBase = id.replace( idRegExp, '' );
if ( ! newIds[ idBase] ) {
newIds[ idBase ] = $form.find( '.siteorigin-widget-input[id^=' + idBase + ']' ).not( '[id*=_id_]' ).length + 1;
}
newId = idBase + '-' + newIds[ idBase ]++;
}
if ( $inputElement.is( '.wp-editor-area' ) ) {
// Prevent potential id overlap by appending the textarea field with a random id.
newId += Math.floor( Math.random() * 1000 );
$inputElement.data( 'tinymce-id', newId );
}
$inputElement.attr( 'id', newId );
if ( $inputElement.is( '.wp-editor-area' ) ) {
var tmceContainer = $inputElement.closest( '.siteorigin-widget-tinymce-container' );
var mediaButtons = tmceContainer.data( 'media-buttons' );
if ( mediaButtons && mediaButtons.html ) {
var idRegExp = new RegExp( id, 'g' );
mediaButtons.html = mediaButtons.html.replace( idRegExp, newId );
tmceContainer.data( 'media-buttons', mediaButtons );
}
}
$copyItem.find( 'label[for=' + id + ']' ).attr( 'for', newId );
$copyItem.find( '[id*=' + id + ']' ).each( function () {
var oldIdAttr = $( this ).attr( 'id' );
var newIdAttr = oldIdAttr.replace( id, newId );
$(this).attr( 'id', newIdAttr );
} );
if (typeof tinymce !== 'undefined' && tinymce.get( newId )) {
tinymce.get( newId ).remove();
}
}
var nestLevel = $item.parents( '.siteorigin-widget-field-repeater' ).length;
var $body = $( 'body' );
if ( ( $body.hasClass( 'wp-customizer' ) || $body.hasClass( 'widgets-php' ) ) && $el.closest( '.panel-dialog' ).length === 0 ) {
nestLevel += 1;
}
var newName = nm.replace( new RegExp( '((?:.*?\\[\\d+\\]){' + ( nestLevel - 1 ).toString() + '})?(.*?\\[)\\d+(\\])' ), '$1$2' + $nextIndex.toString() + '$3' );
$inputElement.attr( 'name', newName );
$inputElement.data( 'original-name', newName );
});
$items.append( $copyItem ).sortable( 'refresh' ).trigger( 'updateFieldPositions' );
$copyItem.sowSetupRepeaterItems();
$copyItem.hide().slideDown( 'fast', function () {
$( window ).trigger( 'resize' );
});
// If increment is enabled for this item, trigger label updates.
var itemLabel = $el.closest( '.siteorigin-widget-field-repeater' ).data( 'item-label' );
if ( typeof itemLabel.increment == 'string' ) {
$el.parent().find( '.siteorigin-widget-field-repeater-item' ).trigger( 'change' )
} else {
$el.trigger( 'change' );
}
if ( isFinite( maxItems ) && $items.find( '.siteorigin-widget-field-repeater-item' ).length == maxItems ) {
$mainRepeater.addClass( 'sow-max-reached' );
}
}
});
$el.find( '> .siteorigin-widget-field-repeater-item-form' ).sowSetupForm();
$el.data( 'sowrepeater-actions-setup', true );
}
});
};
/**
* Strip new lines from a string.
*
* Certain fields, such as the SiteOrigin Editor widget, can contain
* unexpected new lines that can cause issues when comparing values.
*
* @param {*} str The string to strip new lines from.
*
* @returns {string} The string with new lines stripped, or
* the original value if it's not a string.
*/
const stripNewLines = function( str ) {
if ( typeof str !== 'string' || str.length === 0 ) {
return str;
}
return str.replace( /\n/g, '' );
}
/**
* Recursively compares current field values with stored values to determine if they are different.
*
* @param {Object} current - Current field values.
* @param {Object} stored - Stored field values.
* @param {string} key - The key to compare in the current and stored objects.
*
* @returns {boolean} - Returns `true` if the values are different, otherwise `false`.
*/
const fieldBackupCompareLoop = ( current, stored, key ) => {
// Exclude keys that are not relevant for field value comparison.
if (
key === '_sow_form_timestamp' ||
key === '_sow_form_id'
) {
return false;
}
// If the key is missing in the stored values, assume a difference.
if ( ! stored.hasOwnProperty( key ) ) {
return true;
}
const currentValue = current[ key ];
const storedValue = stored[ key ];
// If both values are objects, compare them recursively.
if (
typeof currentValue === 'object' &&
currentValue !== null &&
typeof storedValue === 'object' &&
storedValue !== null
) {
for ( const nestedKey in currentValue ) {
if ( fieldBackupCompareLoop( currentValue, storedValue, nestedKey ) ) {
return true;
}
}
return false;
}
// Compare the values after stripping new lines.
return stripNewLines( currentValue ) !== stripNewLines( storedValue );
};
/**
* Compare current field values with stored values to determine if they are different.
*
* @param {Object} current - Current field values.
* @param {Object} stored - Stored field values.
* @param {string} formId - The ID of the form being compared.
*
* @returns {boolean} - Returns `true` if the current values are different from the stored values, otherwise `false`.
*/
const fieldBackupCompare = ( current, stored, formId ) => {
// Check if the old data is relevant.
if ( current['_sow_form_timestamp'] > stored['_sow_form_timestamp'] ) {
sessionStorage.removeItem( formId );
return false;
}
for ( const key in current ) {
if ( fieldBackupCompareLoop( current, stored, key ) ) {
return true;
}
}
return false;
};
/**
* Handles field backup functionality for Widget Bundle widgets.
*
* This function monitors changes to widget fields and stores
* their values in sessionStorage to allow restoration in case
* of accidental changes or loss. It also displays a notice if
* a newer version of the field values is available for restoration.
*
* @param {jQuery} $mainForm - The main form container for the widget fields.
*/
$.fn.sowFieldBackups = function( $mainForm ) {
const $el = $( this );
const _sow_form_id = $el.find( '> .siteorigin-widgets-form-id' ).val();
const $timestampField = $el.find( '> .siteorigin-widgets-form-timestamp' );
let isUserChange = false;
// Add user change detection to the form to prevent unintended
// backups of automated changes.
$el.on( 'keydown mouseup touchend', ( e ) => {
// Don't trigger a backup if the user clicked on a section.
if ( $( e.target ).parent().is( '.siteorigin-widget-field-type-section' ) ) {
return false;
}
isUserChange = true;
} );
// Debounce backups to prevent potential performance issues.
$el.on( 'change', _.debounce( () => {
if ( isUserChange ) {
$timestampField.val( new Date().getTime() );
const data = sowbForms.getWidgetFormValues( $el );
sessionStorage.setItem( _sow_form_id, JSON.stringify( data ) );
}
}, 500 ) );
// Do we need to show the backup data mismatch notice?
const currentFieldValues = sowbForms.getWidgetFormValues( $el );
if ( ! currentFieldValues ) {
return;
}
// Check if we need to display the field backup notice.
const storedFieldValues = JSON.parse( sessionStorage.getItem( _sow_form_id ) );
if ( ! storedFieldValues ) {
return;
}
// Update the stored timestamp to match the current form's timestamp.
currentFieldValues['_sow_form_timestamp'] = parseInt( $timestampField.val() || 0 );
if ( fieldBackupCompare(
currentFieldValues,
storedFieldValues,
_sow_form_id
) ) {
sowbForms.displayNotice(
$el,
soWidgets.backup.newerVersion,
soWidgets.backup.replaceWarning,
[
{
label: soWidgets.backup.restore,
callback: function ( $notice ) {
sowbForms.setWidgetFormValues(
$mainForm,
storedFieldValues
);
$notice.slideUp( 'fast', function () {
$notice.remove();
} );
},
},
{
label: soWidgets.backup.dismiss,
callback: function ( $notice ) {
$notice.slideUp( 'fast', function () {
sessionStorage.removeItem( _sow_form_id );
$notice.remove();
} );
},
},
]
);
}
}
// Widgets Bundle utility functions
/**
* Get the unique index of a repeated item. Could be in a repeater or if multiple widget fields with the same
* widget class.
*
* @param $el
* @param containerType
* @param containerClass
* @return {*}
*/
sowbForms.getContainerFieldId = function ( $el, containerType, containerClass ) {
var fieldIdPropName = containerType + 'FieldId';
if ( ! this.hasOwnProperty( fieldIdPropName ) ) {
this[ fieldIdPropName ] = 1;
}
var $field = $el.closest( containerClass );
if ( $field.length ) {
var fieldId = $field.data( 'field-id' );
if ( fieldId === undefined ) {
fieldId = this[ fieldIdPropName ]++;
}
$field.data( 'field-id', fieldId );
return fieldId;
}
else {
return false;
}
};
/**
* Retrieve a variable for a field with the given identifier, elementName.
*
* @return {*}
* @param widgetClass The class name of the widget for which to retrieve a variable.
* @param elementName The name of the field for which to retrieve a variable.
* @param key The name of the variable to retrieve.
*/
sowbForms.getWidgetFieldVariable = function (widgetClass, elementName, key) {
var widgetVars = window.sow_field_javascript_variables[widgetClass];
// Get rid of any index placeholders
elementName = elementName.replace(/\[#.*?#\]/g, '');
var variablePath = /[a-zA-Z0-9\-]+(?:\[c?[0-9]+\])?\[(.*)\]/.exec(elementName)[1];
var variablePathParts = variablePath.split('][');
var elementVars = variablePathParts.length ? widgetVars : null;
while (variablePathParts.length) {
elementVars = elementVars[variablePathParts.shift()];
}
return elementVars[key];
};
sowbForms.fetchWidgetVariable = function (key, widget, callback) {
window.sowVars = window.sowVars || {};
if (typeof window.sowVars[widget] === 'undefined') {
$.post(
soWidgets.ajaxurl,
{'action': 'sow_get_javascript_variables', 'widget': widget, 'key': key},
function (result) {
window.sowVars[widget] = result;
callback(window.sowVars[widget][key]);
}
);
}
else {
callback(window.sowVars[widget][key]);
}
};
sowbForms.getWidgetIdBase = function ( formContainer ) {
return formContainer.data( 'id-base' );
};
sowbForms.getWidgetFormValues = function ( formContainer ) {
if ( _.isUndefined( formContainer ) ) {
return null;
}
var data = {};
formContainer.find('*[name]').each(function () {
var $$ = $(this);
try {
var name = /[a-zA-Z0-9\-]+\[[a-zA-Z0-9]+\]\[(.*)\]/.exec( $$.attr( 'name' ) );
if ( _.isEmpty( name ) ) {
return true;
}
// Create an array with the parts of the name
name = name[1];
var parts = name.split( '][' );
// Make sure we either have numbers or strings
parts = parts.map( function ( e ) {
if ( ! isNaN( parseFloat( e ) ) && isFinite( e ) ) {
return parseInt( e );
}
else {
return e;
}
} );
var sub = data;
var fieldValue = null;
var fieldType = _.isString( $$.attr( 'type' ) ) ? $$.attr( 'type' ).toLowerCase() : null;
if ( fieldType === 'checkbox' ) {
if ( $$.is( ':checked' ) ) {
fieldValue = $$.val() !== '' ? $$.val() : true;
} else {
fieldValue = false;
}
} else if ( fieldType === 'radio' ) {
if ( $$.is( ':checked' ) ) {
fieldValue = $$.val();
} else {
return;
}
} else if ( $$.prop( 'tagName' ) === 'TEXTAREA' && $$.hasClass( 'wp-editor-area' ) ) {
// This is a TinyMCE editor, so we'll use the tinyMCE object to get the content
var editor = null;
if ( typeof tinyMCE !== 'undefined' ) {
editor = tinyMCE.get( $$.attr( 'id' ) );
}
if ( editor !== null && typeof( editor.getContent ) === "function" && !editor.isHidden() ) {
fieldValue = editor.getContent();
}
else {
fieldValue = $$.val();
}
} else if ( $$.prop( 'tagName' ) === 'SELECT' ) {
var selected = $$.find( 'option:selected' );
if ( selected.length === 1 ) {
fieldValue = $$.find( 'option:selected' ).val();
} else if ( selected.length > 1 ) {
const selectedOptions = $$.find( 'option:selected' );
fieldValue = [];
for ( var i = 0; i < selectedOptions.length; i++ ) {
fieldValue.push( selectedOptions[ i ].value );
}
}
} else {
fieldValue = $$.val();
}
for ( var i = 0; i < parts.length; i++ ) {
if ( i === parts.length - 1 ) {
if ( parts[i] === '' ) {
// This needs to be an array
sub.push( fieldValue );
} else {
sub[ parts[ i ] ] = fieldValue;
}
}
else {
if ( _.isUndefined( sub[ parts[ i ] ] ) ) {
// We assume that a numeric key means it's an array. (or empty string??)
if ( _.isNumber( parts[ i + 1 ] ) || parts[ i + 1 ] === '' ) {
sub[ parts[ i ] ] = [];
} else {
sub[ parts[ i ] ] = {};
}
}
// Go deeper into the data and continue
sub = sub[ parts[ i ] ];
}
}
} catch ( error ) {
console.error( 'Field [' + $$.attr( 'name' ) + '] could not be processed and was skipped - ' + error.message );
}
});
return data;
};
sowbForms.isEnter = function( e, triggerClick = false ) {
if ( e.which == 13 ) {
if ( triggerClick ) {
$( e.target ).trigger( 'click' );
} else {
return true;
}
}
};
/**
* Sets all the widget form fields in the given container with the given data values.
*
* @param formContainer The jQuery element containing the widget form fields.
* @param data The data from which to set the widget form field values.
* @param skipMissingValues If `true`, this will skip form fields for which the data values are missing.
* If `false`, the form fields will be cleared. Default is `false`.
* @param triggerChange If `true`, trigger a 'change' event on each element after it's value is set. Default is `true`.
*/
sowbForms.setWidgetFormValues = function (formContainer, data, skipMissingValues, triggerChange) {
skipMissingValues = skipMissingValues || false;
triggerChange = (typeof triggerChange !== 'undefined' && triggerChange) || typeof triggerChange === 'undefined';
// First check if this form has any repeaters.
var depth = 0;
var updateRepeaterChildren = function ( formParent, formData ) {
if ( ++depth === 10 ) {
--depth;
return;
}
// Only direct child fields which are repeaters.
formParent.find( '> .siteorigin-widget-field-type-repeater,> .siteorigin-widget-field-type-section > .siteorigin-widget-section > .siteorigin-widget-field-type-repeater' )
.each( function ( index, element ) {
var $this = $( this );
var $repeater = $this.find( '> .siteorigin-widget-field-repeater' );
var repeaterName = $repeater.data( 'repeaterName' );
var repeaterData = formData.hasOwnProperty( repeaterName ) ? formData[ repeaterName ] : null;
var isInSection = $this.parent().is( '.siteorigin-widget-section' );
if ( isInSection ) {
var elementName = $repeater.data( 'element-name' );
// Get rid of any index placeholders
elementName = elementName.replace(/\[#.*?#\]/g, '');
var variablePath = /[a-zA-Z0-9\-]+(?:\[c?[0-9]+\])?\[(.*)\]/.exec(elementName)[1];
var variablePathParts = variablePath.split('][');
var elementVars = variablePathParts.length ? formData : null;
while (variablePathParts.length) {
var key = variablePathParts.shift();
elementVars = elementVars.hasOwnProperty( key ) ? elementVars[ key ] : elementVars;
}
repeaterData = elementVars;
}
if ( ! repeaterData || ! Array.isArray( repeaterData ) ) {
return;
}
// Check that the number of child items matches the number of data items.
var repeaterChildren = $repeater.find( '> .siteorigin-widget-field-repeater-items > .siteorigin-widget-field-repeater-item' );
var numItems = repeaterData.length;
var numChildren = repeaterChildren.length;
if ( numItems > numChildren ) {
// If data items > child items, create extra child items.
for ( var i = 0; i < numItems - numChildren; i++) {
$repeater.find( '> .siteorigin-widget-field-repeater-add' ).trigger( 'click' );
}
} else if ( ! skipMissingValues && numItems < numChildren ) {
// If child items > data items, remove extra child items.
for ( var j = numItems; j < numChildren; j++) {
var $child = $( repeaterChildren.eq( j ) );
$child.find( '> .siteorigin-widget-field-repeater-item-top' )
.find( '.siteorigin-widget-field-remove' )
.trigger( 'click', { silent: true } );
}
}
repeaterChildren = $repeater.find( '> .siteorigin-widget-field-repeater-items > .siteorigin-widget-field-repeater-item' );
for ( var k = 0; k < repeaterChildren.length; k++ ) {
repeaterChildren.eq( k ).find( '> .siteorigin-widget-field-repeater-item-form' );
updateRepeaterChildren(
repeaterChildren.eq( k ).find( '> .siteorigin-widget-field-repeater-item-form' ),
repeaterData[ k ]
);
}
} );
--depth;
};
updateRepeaterChildren(formContainer, data);
$fields = formContainer.find( '*[name]' );
var index = 0;
var validateParts = function( parts ) {
parts.map( function ( e ) {
if ( ! isNaN( parseFloat( e ) ) && isFinite( e ) ) {
return parseInt( e );
} else {
return e;
}
} );
return parts;
};
var getValues = function( data, parts ) {
var sub = data;
var value;
for ( var i = 0; i < parts.length; i++ ) {
// If the field is missing from the data, just leave `value` as `undefined`.
if ( ! sub.hasOwnProperty( parts[ i ] ) ) {
if ( skipMissingValues ) {
continue;
} else {
break;
}
}
if (i === parts.length - 1) {
value = sub[ parts[ i ] ];
} else {
sub = sub[ parts[ i ] ];
}
}
return {
sub: sub,
value: value
};
}
var compareValues = function ( currentValue, newValue ) {
if ( ! newValue ) {
if ( currentValue ) {
return true;
}
} else if ( currentValue !== newValue ) {
return true;
}
return false;
};
var processFields = function( index, $fields ) {
for ( ; index < $fields.length; index++ ) {
if (
index != 0 &&
index + 1 < $fields.length &&
index % 20 == 0
) {
setTimeout( processFields, 150, index + 1, $fields );
return;
}
var $$ = $( $fields[ index ] );
var name = /[a-zA-Z0-9\-]+\[[a-zA-Z0-9]+\]\[(.*)\]/.exec( $$.attr( 'name' ) );
if ( name === undefined || name === null ) {
return true;
}
// There's certain fields we shouldn't process as it can result
// in invalid data, or unintentionally having things processed multiple times.
if (
$$.hasClass( 'sow-measurement-select-unit' ) ||
$$.attr( 'data-presets' ) ||
$$.parent().hasClass( 'siteorigin-widget-field-type-posts' ) ||
(
$$.attr( 'type' ) == 'hidden' &&
! $$.hasClass( 'sow-multi-measurement-input-values' )
)
) {
continue;
}
name = name[1];
var parts = name.split( '][' );
// Make sure we either have numbers or strings
parts = validateParts( parts );
var values = getValues( data, parts )
if ( skipMissingValues && values.value == '' ) {
continue;
}
if ( typeof values.value == 'undefined' ) {
continue;
}
var updated = false;
// This is the end, so we need to set the value on the field here.
if ( $$.attr( 'type' ) === 'checkbox' && $$.is( ':checked' ) != values.value ) {
$$.prop( 'checked', values.value );
updated = true;
} else if ( $$.attr( 'type' ) === 'radio' ) {
$$.prop( 'checked', values.value === $$.val() );
updated = true;
} else if ( $$.prop( 'tagName' ) === 'TEXTAREA' && $$.hasClass( 'wp-editor-area' ) ) {
// This is a TinyMCE editor, so we'll use the tinyMCE object to get the content
var editor = null;
if ( typeof tinyMCE !== 'undefined' ) {
editor = tinyMCE.get( $$.attr( 'id' ) );
}
if ( editor !== null && typeof( editor.setContent ) === "function" && ! editor.isHidden() && $$.parent().is( ':visible' ) ) {
if ( compareValues( editor.getContent(), values.value ) ) {
if ( editor.initialized ) {
editor.setContent( values.value );
updated = true;
} else {
editor.on('init', function () {
editor.setContent( values.value );
});
updated = true;
}
}
} else if ( compareValues( $$.val(), values.value ) ) {
$$.val( values.value );
updated = true;
}
} else if ( $$.is( '.panels-data' ) ) {
if ( compareValues( $$.val(), values.value ) ) {
$$.val( values.value );
var builder = $$.data( 'builder' );
if ( builder ) {
builder.setDataField( $$ );
updated = true;
}
}
} else if ( $$.hasClass( 'sow-multi-measurement-input-values' ) ) {
const $inputs = $$.prev().find( '.sow-multi-measurement-input, .sow-multi-measurement-select-unit' );
const valuesArray = [];
// Only process field if the current editor isn't the Block
// Editor, or it doesn't have a state handler/emitter.
if (
! isBlockEditor &&
(
$$.attr( 'data-state' ) ||
$$.attr( 'data-state-handler' )
)
) {
continue;
}
if ( values.value !== '' ) {
values.value.split(' ').forEach( field => {
valuesArray.push( parseInt( field, 10 ) );
valuesArray.push( field.replace( parseInt( field, 10 ), '' ) );
} );
}
$inputs.each( function( index, element ) {
const $input = $( element );
if ( typeof valuesArray[ index ] !== 'number' ) {
return true;
}
const part = parseInt( valuesArray[ index ] );
$input.val(
isNaN( part ) ? '' : part
);
if ( ! updated && compareValues( $input.val(), values.value[ part ] ) ) {
updated = true;
}
} );
if ( updated ) {
$$.val( values.value );
}
} else if ( compareValues( $$.val(), values.value ) ) {
$$.val( values.value );
updated = true;
}
if ( triggerChange && updated ) {
if (
triggerChange == 'preset' &&
(
! $$.hasClass( 'siteorigin-widget-input-color' ) &&
! $$.hasClass( 'siteorigin-widget-input-slider' ) &&
! $$.is( 'siteorigin-widget-input-select' ) &&
! $$.attr( 'type' ) == 'checkbox'
)
) {
continue;
}
$$.trigger( 'change' );
this.dispatchEvent( new Event( 'change', { bubbles: true, cancelable: true } ) );
}
}
};
processFields( index, $fields );
};
/**
* Displays an informational notice either at the top of the supplied container, or above the optionally supplied
* element.
*
* @param $container The jQuery container in which the notice will be prepended.
* @param title The string title for the notice.
* @param message The string detail message for the notice.
* @param buttons An array of buttons which will be display along with the notice.
* @param $element The optional jQuery element before which the notice will be inserted. If this is supplied it
* will take precedence over the $container argument.
*
*/
sowbForms.displayNotice = function ( $container, title, message, buttons, $element ) {
var $notice = $( '<div class="siteorigin-widget-form-notification"></div>' );
if ( title ) {
$notice.append( '<span>' + title + '</span>' );
}
if ( buttons && buttons.length ) {
buttons.forEach( function ( button ) {
var buttonClasses = '';
if ( button.classes && button.classes.length ) {
buttonClasses = ' ' + button.classes.join( ' ' );
}
var $button = $( '<a class="button button-small' + buttonClasses + '" tabindex="0" target="_blank" rel="noopener noreferrer">' + button.label + '</a>' );
if ( button.url ) {
$button.attr( 'href', button.url );
}
if ( button.callback ) {
$button.on( 'click keyup', function ( e ) {
if ( e.type == 'keyup' && ! sowbForms.isEnter( e ) ) {
return;
}
button.callback( $notice );
});
}
$notice.append( $button );
} );
}
if ( message ) {
$notice.append( '<div><small>' + message + '</small></div>' );
}
if ( $element ) {
$element.before( $notice );
} else {
$container.prepend( $notice );
}
};
/**
* Look for and valid any fields that are required.
*/
sowbForms.validateFields = function( form, showPrompt = true ) {
var valid = true;
var devValidation = $( document ).triggerHandler(
'sow_validate_widget_data',
[
valid,
form,
// Widget ID.
typeof jQuery( '.widget-content' ).data( 'id-base' ) !== undefined ? form.find( '.siteorigin-widget-form' ).data( 'id-base' ) : ''
]
);
if ( typeof devValidation == 'boolean' && ! devValidation ) {
valid = false;
}
if ( valid ) {
var missingRequired = false;
var $so_widgets = form.find( '.siteorigin-widget-field-is-required' );
if ( $so_widgets.length ) {
form.find( '.siteorigin-widget-field-is-required' ).each( function() {
var $$ = $( this );
var $field = $$.find( '.siteorigin-widget-input' );
// Check if this field is inside of a Repeater's HTML clone field.
if ( $field.parents( '.siteorigin-widget-field-repeater-item-html' ).length ) {
return;
}
if (
! $field.val() ||
(
$$.hasClass( 'siteorigin-widget-field-type-checkboxes' ) &&
! $field.prop( 'checked' )
)
) {
missingRequired = true;
$$.addClass( 'sow-required-error' );
}
$field.on( 'change', function( e ) {
$$.removeClass( 'sow-required-error' );
} )
} );
if (
missingRequired &&
(
! showPrompt ||
! confirm( soWidgets.missing_required )
)
) {
valid = false;
}
}
}
return valid;
}
// Validate widget added using Page Builder.
if ( typeof panelsOptions == 'object' ) {
$( document ).on( 'close_dialog_validation', function( e, values, widget, id, instance ) {
return sowbForms.validateFields( $( instance.el ) );
} );
}
// Validate widget added using Classic Widgets & Customizer
$( 'body' ).on( 'click', '.widget-control-save', function( e ) {
var $form = $( this ).parents( '.widget.open' );
if ( $form.length ) {
$form = $form.find( '.widget-content' );
if ( $form.length ) {
if ( ! sowbForms.validateFields( $form ) ) {
e.preventDefault();
e.stopPropagation();
}
}
}
} );
// Further widget validation code for Customizer.
if ( typeof wp != 'undefined' && typeof wp.customize != 'undefined' ) {
jQuery( document ).on( 'widget-added widget-updated widget-synced', function( e, widget, form = false ) {
if ( form.length ) {
sowbForms.validateFields( $( form ) )
}
} );
}
// When we click on a widget top
$('.widgets-holder-wrap').on('click', '.widget:has(.siteorigin-widget-form-main) .widget-top', function () {
var $$ = $(this).closest('.widget').find('.siteorigin-widget-form-main');
setTimeout(function () {
$$.sowSetupForm();
}, 200);
});
// Setup new widgets when they're added in the Customizer or new widgets interface.
$( document ).on( 'widget-added', function( e, widget ) {
widget.find( '.siteorigin-widget-form' ).sowSetupForm();
} );
if ( isBlockEditor ) {
// Setup new widgets when they're previewed in the block editor.
$(document).on('panels_setup_preview', function () {
if (window.hasOwnProperty('sowb')) {
$( sowb ).trigger( 'setup_widgets', { preview: true } );
}
});
}
$( document ).on( 'open_dialog', function ( e, dialog ) {
// When we open a Page Builder edit widget dialog
if ( dialog.$el.find( '.so-panels-dialog' ).is( '.so-panels-dialog-edit-widget' ) ) {
var $fields = dialog.$el.find( '.siteorigin-widget-form-main' ).find( '> .siteorigin-widget-field' );
$fields.trigger( 'sowsetupformfield' );
}
});
$(function () {
$(document).trigger('sowadminloaded');
});
})(jQuery);
var sowEmitters = {
/**
* Find the group/state and an extra match part.
*
* @param arg
* @param matchPart
* @return {*}
*/
'_match': function (arg, matchPart) {
if (typeof matchPart === 'undefined') {
matchPart = '.*';
}
// Create the regular expression to match the group/state and extra match
var exp = new RegExp('^([a-zA-Z0-9_-]+)(\\[([a-zA-Z0-9_-]+)\\])? *: *(' + matchPart + ') *$');
var m = exp.exec(arg);
if (m === null) {
return false;
}
var state = '';
var group = 'default';
if (m[3] !== undefined) {
group = m[1];
state = m[3];
}
else {
state = m[1];
}
return {
'match': m[4].trim(),
'group': group,
'state': state
};
},
'_checker': function (val, args, matchPart, callback) {
var returnStates = {};
if (typeof args.length === 'undefined') {
args = [args];
}
var m;
for (var i = 0; i < args.length; i++) {
m = sowEmitters._match(args[i], matchPart);
if (m === false) {
continue;
}
if (m.match === '_true' || callback(val, args, m.match)) {
returnStates[m.group] = m.state;
}
}
return returnStates;
},
/**
* A very simple state emitter that simply sets the given group the value
*
*
* @param val
* @param args
* @returns {{}}
*/
'select': function (val, args) {
if (typeof args.length === 'undefined') {
args = [args];
}
var returnGroups = {};
for (var i = 0; i < args.length; i++) {
if (args[i] === '') {
args[i] = 'default';
}
returnGroups[args[i]] = val;
}
return returnGroups;
},
/**
* The conditional state emitter uses eval to check a given conditional argument.
*
* @param val
* @param args
* @return {{}}
*/
'conditional': function (val, args) {
return sowEmitters._checker(val, args, '[^;{}]*', function (val, args, match) {
return eval(match);
});
},
/**
* The in state emitter checks if the value is in an array of functions
*
* @param val
* @param args
* @return {{}}
*/
'in': function (val, args) {
return sowEmitters._checker(val, args, '[^;{}]*', function (val, args, match) {
return match.split(',').map(function (s) {
return s.trim();
}).indexOf(val) !== -1;
});
}
};
window.sowbForms = sowbForms;