/* globals easy_wp_smtp, jconfirm, ajaxurl */ 'use strict'; var EasyWPSMTP = window.EasyWPSMTP || {}; EasyWPSMTP.Admin = EasyWPSMTP.Admin || {}; /** * Easy WP SMTP Admin area module. * * @since 2.0.0 */ EasyWPSMTP.Admin.Settings = EasyWPSMTP.Admin.Settings || ( function( document, window, $ ) { /** * Public functions and properties. * * @since 2.0.0 * * @type {object} */ var app = { /** * State attribute showing if one of the plugin settings * changed and was not yet saved. * * @since 2.0.0 * * @type {boolean} */ pluginSettingsChanged: false, /** * Start the engine. DOM is not ready yet, use only to init something. * * @since 2.0.0 */ init: function() { // Do that when DOM is ready. $( app.ready ); }, /** * DOM is fully loaded. * * @since 2.0.0 */ ready: function() { app.pageHolder = $( '.easy-wp-smtp-tab-settings' ); app.settingsForm = $( '.easy-wp-smtp-connection-settings-form' ); // If there are screen options we have to move them. if( $( '#screen-meta-links' ).length > 0 ) { $( '#screen-meta-links, #screen-meta' ).prependTo( '#easy-wp-smtp-header-temp' ); $( '#screen-meta-links' ).show(); } app.bindActions(); app.cleanQueryParams( [ 'sendlayer_quick_connect_result', 'sendlayer_quick_connect_disconnect_result' ] ); app.setJQueryConfirmDefaults(); // Clear Debug Log handler. $( '#easy-wp-smtp-clean-debug-log' ).click( function( e ) { e.preventDefault(); var $btn = $( this ); $.confirm( { backgroundDismiss: false, escapeKey: true, animationBounce: 1, type: 'orange', icon: app.getModalIcon( 'exclamation-triangle-orange' ), title: easy_wp_smtp.heads_up_title, content: easy_wp_smtp.clear_debug_log, buttons: { confirm: { text: easy_wp_smtp.yes_text, btnClass: 'btn-confirm', keys: [ 'enter' ], action: function() { $btn.addClass( 'easy-wp-smtp-btn--loading' ); jQuery.ajax( { url: ajaxurl, type: "post", data: {action: "swpsmtp_clear_log", nonce: easy_wp_smtp.nonce} } ).done( function( data ) { var message, icon, type; if ( data === '1' ) { message = easy_wp_smtp.debug_log_cleared; icon = 'check-circle-green'; type = 'green'; } else { message = easy_wp_smtp.error_occurred + ' ' + data; icon = 'times-circle-red'; type = 'red'; } app.displayAlertModal( message, icon, type ); $btn.removeClass( 'easy-wp-smtp-btn--loading' ); } ); } }, cancel: { text: easy_wp_smtp.cancel_text, btnClass: 'btn-cancel', } } } ); } ); }, /** * Process all generic actions/events, mostly custom that were fired by our API. * * @since 2.0.0 */ bindActions: function() { app.mailers.sendlayer.bindActions(); app.mailers.smtp.bindActions(); app.triggerExitNotice(); app.beforeSaveChecks(); // Open/close meta box. $( '.easy-wp-smtp-meta-box__header' ).on( 'click', function( e ) { // Prevent meta box close/open if link or button was clicked. if ( e.target.tagName === 'A' || e.target.tagName === 'BUTTON' ) { return; } $( this ).closest( '.easy-wp-smtp-meta-box' ).toggleClass( 'easy-wp-smtp-meta-box--closed' ); } ); // Hide all mailers options and display for a currently clicked one. $( '.easy-wp-smtp-mailers-picker__input', app.settingsForm ).on( 'change', function() { $( '.easy-wp-smtp-mailer-options', app.settingsForm ).removeClass( 'easy-wp-smtp-mailer-options--active' ); $( '.easy-wp-smtp-mailer-options[data-mailer="' + $( this ).val() + '"]', app.settingsForm ).addClass( 'easy-wp-smtp-mailer-options--active' ); } ); // Display education modal for mailer if it's disabled. $( '.easy-wp-smtp-mailers-picker__mailer--disabled', app.settingsForm ).on( 'click', function() { var $input = $( this ).prev( '.easy-wp-smtp-mailers-picker__input' ); if ( $input.hasClass( 'easy-wp-smtp-educate' ) ) { app.education.upgradeMailer( $input ); } } ); // Register change event to show/hide plugin supported settings for currently selected mailer. $( '.easy-wp-smtp-mailers-picker__input', app.settingsForm ).on( 'change', this.processMailerSettingsOnChange ); // Show/hide advanced settings. $( '#easy-wp-smtp-setting-advanced', app.settingsForm ).on( 'change', function() { $( this ).closest( '.easy-wp-smtp-row' ).nextAll( '.easy-wp-smtp-row' )[ $( this ).is( ':checked' ) ? 'removeClass' : 'addClass' ]( 'easy-wp-smtp-hidden' ); } ); // Update custom test email fields. $( '#easy-wp-smtp-setting-test_email_custom' ).on( 'change', function() { // Show/hide custom test email fields. $( '#easy-wp-smtp-setting-row-test_email_subject, #easy-wp-smtp-setting-row-test_email_message' ).toggle( $( this ).is( ':checked' ) ); // Add/remove custom test email fields required prop. $( '#easy-wp-smtp-setting-test_email_subject, #easy-wp-smtp-setting-test_email_message' ).prop( 'required', $( this ).is( ':checked' ) ); $( '#easy-wp-smtp-setting-test_email_html' ).prop( 'disabled', $( this ).is( ':checked' ) ); var $html_email = $( '#easy-wp-smtp-setting-test_email_html' ); if ( $( this ).is( ':checked' ) ) { $html_email.data( 'value', $html_email.is( ':checked' ) ); $html_email.prop( 'checked', false ).prop( 'disabled', true ); } else { $html_email.prop( 'checked', $html_email.data( 'value' ) ).prop( 'disabled', false ); } } ); // Dismiss Pro banner at the bottom of the page. $( '.js-easy-wp-smtp-pro-banner-dismiss', app.pageHolder ).on( 'click', function(e) { e.preventDefault(); $.ajax( { url: ajaxurl, dataType: 'json', type: 'POST', data: { action: 'easy_wp_smtp_ajax', task: 'pro_banner_dismiss', nonce: easy_wp_smtp.nonce } } ) .always( function() { $( '.easy-wp-smtp-pro-banner', app.pageHolder ).fadeOut( 'fast' ); } ); } ); // Dissmis educational notices for certain mailers. $( '.js-easy-wp-smtp-mailer-notice-dismiss', app.settingsForm ).on( 'click', function( e ) { e.preventDefault(); var $btn = $( this ), $notice = $btn.parents( '.easy-wp-smtp-notice' ); if ( $btn.hasClass( 'disabled' ) ) { return false; } $.ajax( { url: ajaxurl, dataType: 'json', type: 'POST', data: { action: 'easy_wp_smtp_ajax', nonce: easy_wp_smtp.nonce, task: 'notice_dismiss', notice: $notice.data( 'notice' ), mailer: $notice.data( 'mailer' ) }, beforeSend: function() { $btn.addClass( 'disabled' ); } } ) .always( function() { $notice.fadeOut( 'fast', function() { $btn.removeClass( 'disabled' ); } ); } ); } ); // Show/hide debug output. $( '.easy-wp-smtp-test-email-debug .easy-wp-smtp-error-log-toggle' ).on( 'click', function( e ) { e.preventDefault(); $( '.easy-wp-smtp-test-email-debug .easy-wp-smtp-error-log' ).slideToggle(); } ); // Copy debug output to clipboard. $( '.easy-wp-smtp-test-email-debug .easy-wp-smtp-error-log-copy' ).on( 'click', function( e ) { e.preventDefault(); var $self = $( this ); // Get error log. var $content = $( '.easy-wp-smtp-test-email-debug .easy-wp-smtp-error-log' ); // Copy to clipboard. if ( ! $content.is( ':visible' ) ) { $content.addClass( 'easy-wp-smtp-error-log-selection' ); } var range = document.createRange(); range.selectNode( $content[ 0 ] ); window.getSelection().removeAllRanges(); window.getSelection().addRange( range ); document.execCommand( 'Copy' ); window.getSelection().removeAllRanges(); $content.removeClass( 'easy-wp-smtp-error-log-selection' ); $self.addClass( 'easy-wp-smtp-error-log-copy-copied' ); setTimeout( function() { $self.removeClass( 'easy-wp-smtp-error-log-copy-copied' ); }, 1500 ); } ); // Remove mailer connection. $( '.js-easy-wp-smtp-provider-remove', app.settingsForm ).on( 'click', function() { return confirm( easy_wp_smtp.text_provider_remove ); } ); // Copy input text to clipboard. $( '.easy-wp-smtp-setting-copy', app.settingsForm ).on( 'click', function( e ) { e.preventDefault(); var target = $( '#' + $( this ).data( 'source_id' ) ).get( 0 ); target.select(); document.execCommand( 'Copy' ); var $copyIcon = $( this ).find( 'svg:first-child' ), $checkIcon = $( this ).find( 'svg:last-child' ); $copyIcon.hide(); $checkIcon .show() .fadeOut( 1000, 'swing', function() { $copyIcon.fadeIn( 200 ); } ); } ); // Disable multiple click on the Email Test tab submit button and display a loader icon. $( '.easy-wp-smtp-tab-tools-test #easy-wp-smtp-email-test-form' ).on( 'submit', function() { var $button = $( this ).find( '.easy-wp-smtp-btn' ); $button.attr( 'disabled', true ); $button.addClass( 'easy-wp-smtp-btn--loading' ); } ); // Enable/disable domain check sub options. $( '#easy-wp-smtp-setting-domain_check' ).on( 'change', function() { $( '#easy-wp-smtp-setting-domain_check_allowed_domains, #easy-wp-smtp-setting-domain_check_do_not_send' ).prop( 'disabled', ! $( this ).is( ':checked' ) ) } ); // Obfuscated fields $( '.easy-wp-smtp-btn[data-clear-field]' ).on( 'click', function( e ) { var $button = $( this ); var fieldId = $button.attr( 'data-clear-field' ); var $field = $( `#${fieldId}` ); $field.prop( 'disabled', false ); $field.attr( 'name', $field.attr( 'data-name' ) ); $field.removeAttr( 'value' ); $field.focus(); $button.remove(); } ); $( '#easy-wp-smtp-setting-rate_limit-lite' ).on( 'click', function( e ) { e.preventDefault(); app.education.rateLimitUpgrade(); } ); }, education: { upgradeModal: function( title, content, upgradeUrlUtmContent ) { $.alert( { backgroundDismiss: true, escapeKey: true, animationBounce: 1, type: 'blue', closeIcon: true, title: title, icon: '">' + easy_wp_smtp.education.upgrade_icon_lock + '' + easy_wp_smtp.education.upgrade_doc + '' ); this.$body.addClass( 'easy-wp-smtp-upgrade-mailer-education-modal' ); }, buttons: { confirm: { text: easy_wp_smtp.education.upgrade_button, btnClass: 'easy-wp-smtp-btn easy-wp-smtp-btn--green', keys: [ 'enter' ], action: function() { var appendChar = /(\?)/.test( easy_wp_smtp.education.upgrade_url ) ? '&' : '?', upgradeURL = easy_wp_smtp.education.upgrade_url + appendChar + 'utm_content=' + encodeURIComponent( upgradeUrlUtmContent ); window.open( upgradeURL, '_blank' ); } } } } ); }, upgradeMailer: function( $input ) { this.upgradeModal( easy_wp_smtp.education.upgrade_title.replace( /%name%/g, $input.data( 'title' ) ), easy_wp_smtp.education.upgrade_content.replace( /%name%/g, $input.data( 'title' ) ) + easy_wp_smtp.education.upgrade_bonus, $input.val() ); }, rateLimitUpgrade: function() { this.upgradeModal( easy_wp_smtp.education.rate_limit.upgrade_title, easy_wp_smtp.education.rate_limit.upgrade_content + easy_wp_smtp.education.upgrade_bonus, 'rate-limit-setting' ); }, }, /** * Individual mailers specific js code. * * @since 2.0.0 */ mailers: { sendlayer: { /** * Show a SendLayer connect error modal with message and optional error code. * * @since 2.14.0 * * @param {string} message The error message to display. * @param {string} errorCode The dot-notation error code (optional). */ showConnectError: function( message, errorCode ) { var content = '

' + $( '' ).text( message ).html() + '

'; if ( errorCode ) { content += '
' + '' + $( '' ).text( errorCode ).html() + '' + '' + '
'; } $.alert( { backgroundDismiss: true, escapeKey: true, animationBounce: 1, type: 'red', closeIcon: true, icon: app.getModalIcon( 'times-circle-red' ), title: easy_wp_smtp.sendlayer.error_title, content: content, boxWidth: '450px', buttons: { confirm: { text: easy_wp_smtp.ok_text, btnClass: 'easy-wp-smtp-btn easy-wp-smtp-btn-md', keys: [ 'enter' ] } }, onOpenBefore: function() { this.$body.on( 'click', '.easy-wp-smtp-error-code-box__copy', function() { var $btn = $( this ); var code = $btn.siblings( 'code' ).text(); if ( navigator.clipboard ) { navigator.clipboard.writeText( code ); } $btn.find( '.easy-wp-smtp-error-code-box__icon-copy' ).hide(); $btn.find( '.easy-wp-smtp-error-code-box__icon-check' ).show(); setTimeout( function() { $btn.find( '.easy-wp-smtp-error-code-box__icon-check' ).hide(); $btn.find( '.easy-wp-smtp-error-code-box__icon-copy' ).show(); }, 2000 ); } ); } } ); }, /** * Start the connect flow via AJAX and handle errors with the modal. * * @since 2.14.0 * * @param {Object} connectArgs Extra arguments to pass to the connect endpoint (e.g. { utm_content: '...' }). * @param {Function} onDone Callback when the request completes (success or error). */ doConnect: function( connectArgs, onDone ) { var self = this; var returnUrl = $( '#easy-wp-smtp-sendlayer-quick-connect-return-url' ).val() || easy_wp_smtp.sendlayer.return_url; var connectionId = $( '#easy-wp-smtp-sendlayer-quick-connect-connection-id' ).val() || ''; $.post( ajaxurl, { action: 'easy_wp_smtp_sendlayer_connect', nonce: easy_wp_smtp.sendlayer.connect_nonce, return_url: returnUrl, connection_id: connectionId, connect_args: connectArgs || {}, }, function( response ) { if ( response.success && response.data.redirect_url ) { window.location.href = response.data.redirect_url; } else { var message = response.data && response.data.message ? response.data.message : easy_wp_smtp.sendlayer.error_text; var errorCode = response.data && response.data.error_code ? response.data.error_code : ''; self.showConnectError( message, errorCode ); if ( onDone ) { onDone(); } } } ).fail( function() { self.showConnectError( easy_wp_smtp.sendlayer.server_error, 'plugin.init_connect.ajax_failed' ); if ( onDone ) { onDone(); } } ); }, /** * Bind SendLayer-specific UI actions. * * @since 2.14.0 */ bindActions: function() { var self = this; // Quick Connect button. $( '#easy-wp-smtp-sendlayer-connect-btn' ).on( 'click', function( e ) { e.preventDefault(); var $btn = $( this ); $btn.addClass( 'easy-wp-smtp-btn--loading' ); self.doConnect( { utm_content: 'Plugin Settings - Quick Connect' }, function() { $btn.removeClass( 'easy-wp-smtp-btn--loading' ); } ); } ); // Change domain link (same flow as Quick Connect). $( '#easy-wp-smtp-sendlayer-change-domain' ).on( 'click', function( e ) { e.preventDefault(); var $link = $( this ); var originalText = $link.text(); $link.text( easy_wp_smtp.sendlayer.connecting_text ); self.doConnect( { utm_content: 'Plugin Settings - Quick Connect Change Domain' }, function() { $link.text( originalText ); } ); } ); // Show API key field and remove the toggle link. $( '#easy-wp-smtp-sendlayer-show-api-key' ).on( 'click', function( e ) { e.preventDefault(); $( this ).closest( '.easy-wp-smtp-setting-row' ).remove(); $( '#easy-wp-smtp-setting-row-sendlayer-api_key' ).show(); } ); // SendLayer education banner: Setup button (same flow as Quick Connect). $( '#easy-wp-smtp-sendlayer-education-connect-btn' ).on( 'click', function( e ) { e.preventDefault(); var $btn = $( this ); $btn.addClass( 'easy-wp-smtp-btn--loading' ); self.doConnect( { utm_content: 'Plugin Settings - Quick Connect Education' }, function() { $btn.removeClass( 'easy-wp-smtp-btn--loading' ); } ); } ); // SendLayer education banner: Dismiss. $( '.js-easy-wp-smtp-sendlayer-education-dismiss' ).on( 'click', function( e ) { e.preventDefault(); var $banner = $( this ).closest( '.easy-wp-smtp-sendlayer-education' ); $banner.fadeOut( 200 ); $.post( ajaxurl, { action: 'easy_wp_smtp_ajax', task: 'notice_dismiss', notice: 'sendlayer_education', nonce: easy_wp_smtp.nonce, } ); } ); } }, smtp: { bindActions: function() { // Hide SMTP-specific user/pass when Auth disabled. $( '#easy-wp-smtp-setting-smtp-auth' ).on( 'change', function() { $( '#easy-wp-smtp-setting-row-smtp-user, #easy-wp-smtp-setting-row-smtp-pass' ).toggleClass( 'easy-wp-smtp-hidden' ); } ); // Port default values based on encryption type. $( '#easy-wp-smtp-setting-row-smtp-encryption input' ).on( 'change', function() { var $input = $( this ), $smtpPort = $( '#easy-wp-smtp-setting-smtp-port', app.settingsForm ); if ( 'tls' === $input.val() ) { $smtpPort.val( '587' ); $( '#easy-wp-smtp-setting-row-smtp-autotls' ).addClass( 'easy-wp-smtp-hidden' ); } else if ( 'ssl' === $input.val() ) { $smtpPort.val( '465' ); $( '#easy-wp-smtp-setting-row-smtp-autotls' ).removeClass( 'easy-wp-smtp-hidden' ); } else { $smtpPort.val( '25' ); $( '#easy-wp-smtp-setting-row-smtp-autotls' ).removeClass( 'easy-wp-smtp-hidden' ); } } ); } } }, /** * Exit notice JS code when plugin settings are not saved. * * @since 2.0.0 */ triggerExitNotice: function() { var $settingPages = $( '.easy-wp-smtp-page-general' ); // Display an exit notice, if settings are not saved. $( window ).on( 'beforeunload', function() { if ( app.pluginSettingsChanged ) { return easy_wp_smtp.text_settings_not_saved; } } ); // Set settings changed attribute, if any input was changed. $( ':input:not( #easy-wp-smtp-setting-license-key, .easy-wp-smtp-not-form-input, #easy-wp-smtp-setting-outlook-one_click_setup_enabled )', $settingPages ).on( 'change', function() { app.pluginSettingsChanged = true; } ); // Clear the settings changed attribute, if the settings are about to be saved. $( 'form', $settingPages ).on( 'submit', function() { app.pluginSettingsChanged = false; } ); }, /** * Perform any checks before the settings are saved. * * Checks: * - warn users if they try to save the settings with the default (PHP) mailer selected. * * @since 2.0.0 */ beforeSaveChecks: function() { app.settingsForm.on( 'submit', function() { if ( $( '.easy-wp-smtp-mailers-picker__input:checked', app.settingsForm ).val() === 'mail' ) { var $thisForm = $( this ); $.alert( { backgroundDismiss: false, escapeKey: false, animationBounce: 1, type: 'orange', icon: app.getModalIcon( 'exclamation-triangle-orange' ), title: easy_wp_smtp.default_mailer_notice.title, content: easy_wp_smtp.default_mailer_notice.content, boxWidth: '550px', buttons: { confirm: { text: easy_wp_smtp.default_mailer_notice.save_button, btnClass: 'btn-confirm', keys: [ 'enter' ], action: function() { $thisForm.off( 'submit' ).trigger( 'submit' ); } }, cancel: { text: easy_wp_smtp.default_mailer_notice.cancel_button, btnClass: 'btn-cancel', }, } } ); return false; } } ); }, /** * On change callback for showing/hiding plugin supported settings for currently selected mailer. * * @since 2.0.0 */ processMailerSettingsOnChange: function() { var selectedMailer = $( this ).val(); var mailerSupportedSettings = easy_wp_smtp.all_mailers_supports[ selectedMailer ]; for ( var setting in mailerSupportedSettings ) { // eslint-disable-next-line no-prototype-builtins if ( mailerSupportedSettings.hasOwnProperty( setting ) ) { $( '.js-easy-wp-smtp-setting-' + setting, app.settingsForm ).toggle( mailerSupportedSettings[ setting ] ); } } // Special case: "from email" (group settings). var $mainSettingInGroup = $( '.js-easy-wp-smtp-setting-from_email' ); var $quickConnectFromEmail = $( '#easy-wp-smtp-setting-row-sendlayer-quick-connect-from_email' ); var isQuickConnectActive = selectedMailer === 'sendlayer' && $quickConnectFromEmail.length > 0; $mainSettingInGroup.closest( '.easy-wp-smtp-setting-row' ).toggle( ! isQuickConnectActive && ( mailerSupportedSettings[ 'from_email' ] || mailerSupportedSettings[ 'from_email_force' ] ) ); // Toggle quick connect From Email field and disable inputs when hidden // to prevent split fields from being submitted for other mailers. $quickConnectFromEmail.toggle( isQuickConnectActive ); $quickConnectFromEmail.find( 'input' ).prop( 'disabled', ! isQuickConnectActive ); // Special case: "from name" (group settings). $mainSettingInGroup = $( '.js-easy-wp-smtp-setting-from_name' ); $mainSettingInGroup.closest( '.easy-wp-smtp-setting-row' ).toggle( mailerSupportedSettings[ 'from_name' ] || mailerSupportedSettings[ 'from_name_force' ] ); // Special case: "return path" (group settings). $mainSettingInGroup = $( '.js-easy-wp-smtp-setting-return-path' ); $mainSettingInGroup.closest( '.easy-wp-smtp-setting-row' ).toggle( !!mailerSupportedSettings[ 'return_path' ] ); }, /** * Set jQuery-Confirm default options. * * @since 2.0.0 */ setJQueryConfirmDefaults: function() { jconfirm.defaults = { typeAnimated: false, draggable: false, animateFromElement: false, theme: 'modern', boxWidth: '450px', useBootstrap: false }; }, /** * Display the modal with provided text and icon. * * @since 2.0.0 * * @param {string} message The message to be displayed in the modal. * @param {string} icon The icon name from /assets/images/icons/ to be used in modal. * @param {string} type The type of the message (red, green, orange, blue, purple, dark). * @param {Function} actionCallback The action callback function. */ displayAlertModal: function( message, icon, type, actionCallback = undefined ) { type = type || 'default'; actionCallback = actionCallback || function() { }; $.alert( { backgroundDismiss: true, escapeKey: true, animationBounce: 1, type: type, closeIcon: true, title: false, icon: icon ? app.getModalIcon( icon ) : '', content: message, buttons: { confirm: { text: easy_wp_smtp.ok_text, btnClass: 'easy-wp-smtp-btn easy-wp-smtp-btn-md', keys: [ 'enter' ], action: actionCallback } } } ); }, /** * Remove transient query params from the URL without a page reload. * * Useful for cleaning up one-time result params after they have been * read and rendered on the current page load. * * @since 2.14.0 * * @param {string[]} params List of query parameter names to remove. */ cleanQueryParams: function( params ) { try { var url = new URL( window.location.href ); var dirty = false; params.forEach( function( param ) { if ( url.searchParams.has( param ) ) { url.searchParams.delete( param ); dirty = true; } } ); if ( dirty ) { window.history.replaceState( {}, document.title, url.toString() ); } } catch ( e ) {} }, /** * Returns prepared modal icon. * * @since 2.0.0 * * @param {string} icon The icon name from /assets/images/icons/ to be used in modal. * * @returns {string} Modal icon HTML. */ getModalIcon: function( icon ) { return '">