HEX
Server: LiteSpeed
System: Linux php-prod-1.spaceapp.ru 5.15.0-157-generic #167-Ubuntu SMP Wed Sep 17 21:35:53 UTC 2025 x86_64
User: xnsbb3110 (1041)
PHP: 8.1.33
Disabled: NONE
Upload Files
File: //proc/self/cwd/wp-content/plugins/wp-smushit/_src/js/common/media-library-scanner.js
/* global WP_Smush */

/**
 * Abstract Media Library Scanner.
 *
 */
import Fetcher from '../utils/fetcher';
import { scanProgressBar } from './progressbar';
import { GlobalStats } from './globalStats';
import loopbackTester from '../loopback-tester';
const { __ } = wp.i18n;
export default class MediaLibraryScanner {
	constructor() {
		this.autoSyncDuration = 1500;
		this.progressTimeoutId = 0;
		this.scanProgress = scanProgressBar( this.autoSyncDuration );
	}

	startScan( optimizeOnScanCompleted = false ) {
		this.onStart();
		const processType = optimizeOnScanCompleted ? 'smush' : 'scan';
		loopbackTester.performTest().then( ( res ) => {
			const isLoopbackHealthy = res?.loopback;
			if ( isLoopbackHealthy ) {
				Fetcher.scanMediaLibrary.start( optimizeOnScanCompleted ).then( ( response ) => {
					if ( ! response?.success ) {
						this.showFailureNotice( response );
						this.onStartFailure( response );
						return;
					}
					this.showProgressBar().autoSyncStatus();
				} );
			} else {
				this.showLoopbackErrorModal( processType );
				this.onStartFailure( res );
			}
		} ).catch( ( error ) => {
			console.error( 'Error:', error );
			this.showLoopbackErrorModal( processType );
			this.onStartFailure( error );
		} );
	}

	onStart() {
		// Do nothing at the moment.
	}

	onStartFailure( response ) {
		// Do nothing at the moment.
	}

	showFailureNotice( response ) {
		WP_Smush.helpers.showNotice( response, {
			showdismiss: true,
			autoclose: false,
		} );
	}

	showLoopbackErrorModal( processType ) {
		const loopbackErrorModal = document.getElementById( 'smush-loopback-error-dialog' );
		if ( ! loopbackErrorModal || ! window.SUI ) {
			return;
		}

		// Cache current process type.
		loopbackErrorModal.dataset.processType = processType || 'scan';

		WP_Smush.helpers.showModal( loopbackErrorModal.id );
	}

	showProgressBar() {
		this.onShowProgressBar();
		this.scanProgress.reset().setOnCancelCallback( this.showStopScanningModal.bind( this ) ).open();
		return this;
	}

	onShowProgressBar() {
		// Do nothing at the moment.
	}

	showStopScanningModal() {
		if ( ! window.SUI ) {
			return;
		}

		this.onShowStopScanningModal();

		window.SUI.openModal(
			'smush-stop-scanning-dialog',
			'wpbody-content',
			undefined,
			false
		);
	}

	onShowStopScanningModal() {
		this.registerCancelProcessEvent();
	}

	registerCancelProcessEvent() {
		const stopScanButton = document.querySelector( '.smush-stop-scanning-dialog-button' );
		if ( ! stopScanButton ) {
			return;
		}

		stopScanButton.addEventListener( 'click', this.cancelProgress.bind( this ), { once: true } );
	}

	closeStopScanningModal() {
		if ( ! window.SUI ) {
			return;
		}
		const stopScanningElement = document.querySelector( '#smush-stop-scanning-dialog' );
		const isModalClosed = ( ! stopScanningElement ) || ! stopScanningElement.classList.contains( 'sui-content-fade-in' );
		if ( isModalClosed ) {
			return;
		}
		window.SUI.closeModal( 'smush-stop-scanning-dialog' );
	}

	closeProgressBar() {
		this.onCloseProgressBar();
		this.scanProgress.close();
	}

	onCloseProgressBar() {
		// Do nothing at the moment.
	}

	updateProgress( stats ) {
		const totalItems = this.getTotalItems( stats );
		const processedItems = this.getProcessedItems( stats );

		return this.scanProgress.update( processedItems, totalItems );
	}

	getProcessedItems( stats ) {
		return stats?.processed_items || 0;
	}

	getTotalItems( stats ) {
		return stats?.total_items || 0;
	}

	cancelProgress() {
		this.scanProgress.setCancelButtonOnCancelling();
		return Fetcher.scanMediaLibrary.cancel().then( ( response ) => {
			if ( ! response?.success ) {
				this.onCancelFailure( response );
				return;
			}
			this.onCancelled( response.data );
		} );
	}

	onCancelFailure( response ) {
		WP_Smush.helpers.showNotice( response, {
			showdismiss: true,
			autoclose: false,
		} );
		this.scanProgress.resetCancelButtonOnFailure();
	}

	onDead( stats ) {
		this.clearProgressTimeout();
		this.closeProgressBar();
		this.closeStopScanningModal();
		this.showRetryScanModal();
	}

	showRetryScanModal() {
		const retryScanModalElement = document.getElementById( 'smush-retry-scan-notice' );
		if ( ! window.SUI || ! retryScanModalElement ) {
			return;
		}

		retryScanModalElement.querySelector( '.smush-retry-scan-notice-button' ).addEventListener( 'click', ( e ) => {
			window.SUI.closeModal( 'smush-retry-scan-notice' );
			const recheckImagesBtn = document.querySelector( '.wp-smush-scan' );
			if ( ! recheckImagesBtn ) {
				return;
			}

			e.preventDefault();
			recheckImagesBtn.click();
		}, { once: true } );

		window.SUI.openModal(
			'smush-retry-scan-notice',
			'wpbody-content',
			undefined,
			false
		);
	}

	onCompleted( stats ) {
		this.onFinish( stats );
	}

	onCancelled( stats ) {
		this.onFinish( stats );
	}

	onFinish( stats ) {
		this.clearProgressTimeout();
		const globalStats = stats?.global_stats;
		this.updateGlobalStatsAndBulkContent( globalStats );
		this.closeProgressBar();
		this.closeStopScanningModal();
	}

	clearProgressTimeout() {
		if ( this.progressTimeoutId ) {
			clearTimeout( this.progressTimeoutId );
		}
	}

	updateGlobalStatsAndBulkContent( globalStats ) {
		if ( ! globalStats ) {
			return;
		}
		GlobalStats.updateGlobalStatsFromSmushScriptData( globalStats );
		GlobalStats.renderStats();
	}

	getStatus() {
		return Fetcher.scanMediaLibrary.getScanStatus();
	}

	autoSyncStatus() {
		const startTime = new Date().getTime();
		this.getStatus().then( ( response ) => {
			if ( ! response?.success ) {
				return;
			}
			const stats = response.data;

			if ( stats.is_dead ) {
				this.onDead( response.data );
				return;
			}

			this.beforeUpdateStatus( stats );

			this.updateProgress( stats ).then( () => {
				this.scanProgress.increaseDurationToHaveChangeOnProgress( new Date().getTime() - startTime );

				const isCompleted = stats?.is_completed;
				if ( isCompleted ) {
					this.onCompleted( stats );
					return;
				}
				const isCancelled = stats?.is_cancelled;
				if ( isCancelled ) {
					this.onCancelled( stats );
					return;
				}

				this.progressTimeoutId = setTimeout( () => this.autoSyncStatus(), this.autoSyncDuration );
			} );
		} );
	}

	beforeUpdateStatus() {
		// Do nothing at the moment.
	}

	setInnerText( element, newText ) {
		if ( ! element ) {
			return;
		}
		element.dataset.originalText = element.dataset.originalText || element.innerText.trim();
		element.innerText = newText;
	}

	revertInnerText( element ) {
		if ( ! element || ! element.dataset.originalText ) {
			return;
		}
		element.innerText = element.dataset.originalText.trim();
	}

	hideAnElement( element ) {
		if ( element ) {
			element.classList.add( 'sui-hidden' );
		}
	}

	showAnElement( element ) {
		if ( element ) {
			element.classList.remove( 'sui-hidden' );
		}
	}
}