You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

273 lines
10 KiB

10 months ago
( function ( $, rwmb, i18n ) {
'use strict';
/**
* Extract the validation key from an input's name attribute. Usually it's the field ID, but sometimes (like for `file`), it's the field's input name.
*
* field[] => field // Fields with multiple values: file, checkbox list, etc.
* field[1] => field // Cloneable fields
* field[1][] => field // Cloneable fields with multiple values: file, checkbox list, etc.
*
* group[field][] => field // Group with fields with multiple values: file, checkbox list, etc.
* group[field][1] => field // Group with cloneable fields
* group[field][1][] => field // Group with cloneable fields with multiple values: file, checkbox list, etc.
*
* group[1][field][] => field // Cloneable group with fields with multiple values: file, checkbox list, etc.
* group[1][field][1] => field // Cloneable group with cloneable fields
* group[1][field][1][] => field // Cloneable group with cloneable fields with multiple values: file, checkbox list, etc.
*
* group[subgroup][field][] => field // Subgroup with fields with multiple values: file, checkbox list, etc.
* group[subgroup][field][1] => field // Subgroup with cloneable fields
* group[subgroup][field][1][] => field // Subgroup with cloneable fields with multiple values: file, checkbox list, etc.
*
* group[subgroup][1][field][] => field // Cloneable subgroup with fields with multiple values: file, checkbox list, etc.
* group[subgroup][1][field][1] => field // Cloneable subgroup with cloneable fields
* group[subgroup][1][field][1][] => field // Cloneable subgroup with cloneable fields with multiple values: file, checkbox list, etc.
*
* group[1][subgroup][field][] => field // Cloneable group with subgroup with fields with multiple values: file, checkbox list, etc.
* group[1][subgroup][field][1] => field // Cloneable group with subgroup with cloneable fields
* group[1][subgroup][field][1][] => field // Cloneable group with subgroup with cloneable fields with multiple values: file, checkbox list, etc.
*
* group[1][subgroup][1][field][] => field // Cloneable group with cloneable subgroup with fields with multiple values: file, checkbox list, etc.
* group[1][subgroup][1][field][1] => field // Cloneable group with cloneable subgroup with cloneable fields
* group[1][subgroup][1][field][1][] => field // Cloneable group with cloneable subgroup with cloneable fields with multiple values: file, checkbox list, etc.
*/
const getValidationKey = name => {
// Detect name parts in format of anything[] or anything[1].
let parts = name.match( /^(.+?)(?:\[\d+\]|(?:\[\]))?$/ );
if ( parts[ 1 ] && isNaN( parts[ 1 ] ) ) {
// Remove []
let words = name.match( /(\w+)|(\[\w+\])/g );
let resultArray = [ words.join( "" ) ];
// Remove characters "[" and "]".
words.forEach( matchedValue => {
if ( matchedValue.startsWith( "[" ) ) {
resultArray.push( matchedValue.substring( 1, matchedValue.length - 1 ) );
} else {
resultArray.push( matchedValue );
}
} );
parts[ 0 ] = resultArray[ 0 ];
parts[ 1 ] = isNaN( resultArray[ resultArray.length - 1 ] ) ? resultArray[ resultArray.length - 1 ] : resultArray[ resultArray.length - 2 ];
}
return parts.pop();
};
/**
* Fix validation not working for cloneable files or fields in groups.
*/
$.validator.staticRules = function ( element ) {
let rules = {},
validator = $.data( element.form, "validator" );
// No rules.
if ( validator.settings.rules === null || Object.keys( validator.settings.rules ).length === 0 ) {
return rules;
}
// Do not validate hidden fields.
if ( element.type === 'hidden' ) {
return rules;
}
let key = getValidationKey( element.name );
/**
* Cloneable files or files in groups.
* Input name is transformed into format `_file_{unique_id}`
* There is also a hidden input with name `_index_{field_id}` with value `_file_{unique_id}`
*
* In this case, `key` is always `_file_{unique_id}`
*
* Note that for cloneable files, validation rule is set for `_index_{field_id}`. For files in groups, validation rule is still `{field_id}`.
*/
if ( element.type === 'file' && ( $( element ).closest( '.rwmb-clone' ).length > 0 || $( element ).closest( '.rwmb-group-wrapper' ).length > 0 ) ) {
const $input = $( element ).closest( '.rwmb-input' );
const $indexInput = $input.find( '*[value="' + key + '"]' );
key = getValidationKey( $indexInput.attr( 'name' ) );
// Remove prefix `_index_` from input name when in groups.
if ( !validator.settings.rules[ key ] && key.includes( '_index_' ) ) {
key = key.slice( 7 );
}
if ( validator.settings.rules[ key ] ) {
// Set message for element.
validator.settings.messages[ element.name ] = validator.settings.messages[ key ];
// Set rule for element.
return $.validator.normalizeRule( validator.settings.rules[ key ] ) || {};
}
return rules;
}
// For normal fields and fields in groups: set rules by their field IDs (validation keys).
// Set message for element.
validator.settings.messages[ element.name ] = validator.settings.messages[ key ];
// Set rule for element.
return $.validator.normalizeRule( validator.settings.rules[ key ] ) || {};
};
/**
* Make jQuery Validation works with multiple inputs with same names.
* Need for file, image fields where users can upload multiple files with same input names.
*
* @link https://stackoverflow.com/q/931687/371240
*/
$.validator.prototype.checkForm = function () {
this.prepareForm();
for ( var i = 0, elements = ( this.currentElements = this.elements() ); elements[ i ]; i++ ) {
if ( this.findByName( elements[ i ].name ).length !== undefined && this.findByName( elements[ i ].name ).length > 1 ) {
for ( var cnt = 0; cnt < this.findByName( elements[ i ].name ).length; cnt++ ) {
this.check( this.findByName( elements[ i ].name )[ cnt ] );
}
} else {
this.check( elements[ i ] );
}
}
return this.valid();
};
class Validation {
constructor( formSelector ) {
this.$form = $( formSelector );
this.validationElements = this.$form.find( '.rwmb-validation' );
this.showAsterisks();
this.getSettings();
}
init() {
this.$form
// Update underlying textarea before submit.
// Don't use submitHandler() because form can be submitted via Ajax on the front end.
.on( 'submit', function () {
if ( typeof tinyMCE !== 'undefined' ) {
tinyMCE.triggerSave();
}
} )
.validate( this.settings );
}
showAsterisks() {
this.validationElements.each( function () {
const data = $( this ).data( 'validation' );
$.each( data.rules, function ( k, v ) {
if ( !v[ 'required' ] ) {
return;
}
let $el = $( '[name="' + k + '"]' );
if ( !$el.length ) {
$el = $( '[name*="[' + k + ']"]' ); // Subfields in groups.
}
if ( $el.length ) {
$el.closest( '.rwmb-input' ).siblings( '.rwmb-label' ).find( 'label' ).append( '<span class="rwmb-required">*</span>' );
}
} );
} );
}
getSettings() {
this.settings = {
ignore: ':not(.rwmb-media,.rwmb-image_select,.rwmb-wysiwyg,.rwmb-color,.rwmb-map,.rwmb-osm,.rwmb-switch,[class|="rwmb"])',
errorPlacement: function ( error, element ) {
error.appendTo( element.closest( '.rwmb-input' ) );
},
errorClass: 'rwmb-error',
errorElement: 'p',
invalidHandler: this.invalidHandler.bind( this )
};
// Gather all validation rules.
var that = this;
this.validationElements.each( function () {
$.extend( true, that.settings, $( this ).data( 'validation' ) );
} );
}
invalidHandler() {
this.showMessage();
// Group field will automatically expand and show an error warning when collapsing
for ( var i = 0; i < this.$form.data( 'validator' ).errorList.length; i++ ) {
$( '#' + this.$form.data( 'validator' ).errorList[ i ].element.id ).closest( '.rwmb-group-collapsed' ).removeClass( 'rwmb-group-collapsed' );
}
// Custom event for showing error fields inside tabs/hidden divs. Use setTimeout() to run after error class is added to inputs.
var that = this;
setTimeout( function () {
that.$form.trigger( 'after_validate' );
}, 200 );
}
showMessage() {
// Re-enable the submit ( publish/update ) button and hide the ajax indicator
$( '#publish' ).removeClass( 'button-primary-disabled' );
$( '#ajax-loading' ).attr( 'style', '' );
$( '#rwmb-validation-message' ).remove();
this.$form.before( '<div id="rwmb-validation-message" class="notice notice-error is-dismissible"><p>' + i18n.message + '</p></div>' );
}
};
class GutenbergValidation extends Validation {
init() {
var that = this,
editor = wp.data.dispatch( 'core/editor' ),
savePost = editor.savePost; // Reference original method.
if ( that.settings ) {
that.$form.validate( that.settings );
}
// Change the editor method.
editor.savePost = function ( options = {} ) {
// Bypass the validation when previewing in Gutenberg.
if ( typeof options === 'object' && options.isPreview ) {
return savePost( options );
}
// Must call savePost() here instead of in submitHandler() because the form has inline onsubmit callback.
if ( that.$form.valid() ) {
return savePost( options );
}
};
}
showMessage() {
wp.data.dispatch( 'core/notices' ).createErrorNotice( i18n.message, {
id: 'meta-box-validation',
isDismissible: true
} );
}
};
// Run on document ready.
function init() {
if ( rwmb.isGutenberg ) {
var advanced = new GutenbergValidation( '.metabox-location-advanced' ),
normal = new GutenbergValidation( '.metabox-location-normal' ),
side = new GutenbergValidation( '.metabox-location-side' );
side.init();
normal.init();
advanced.init();
return;
}
// Edit post, edit term, edit user, front-end form.
var $forms = $( '#post, #edittag, #your-profile, .rwmb-form' );
$forms.each( function () {
var form = new Validation( this );
form.init();
} );
};
rwmb.$document
.on( 'mb_ready', init );
} )( jQuery, rwmb, rwmbValidation );