AlkantarClanX12

Your IP : 3.137.173.98


Current Path : /home/thanudqk/public_html/wp-content/plugins/tablepress/admin/js/import/
Upload File :
Current File : /home/thanudqk/public_html/wp-content/plugins/tablepress/admin/js/import/screen.jsx

/**
 * JavaScript code for the "Import Screen" component.
 *
 * @package TablePress
 * @subpackage Import Screen
 * @author Tobias Bäthge
 * @since 2.2.0
 */

/* globals tp */

/**
 * WordPress dependencies.
 */
import { useEffect, useRef, useState } from 'react';
import {
	Button,
	__experimentalHStack as HStack, // eslint-disable-line @wordpress/no-unsafe-wp-apis
	Icon,
	RadioControl,
	ComboboxControl,
	Disabled,
	TextareaControl,
	TextControl,
} from '@wordpress/components';
import { info } from '@wordpress/icons';
import { __, _n, _x, sprintf } from '@wordpress/i18n';

// Details for the available import sources.
const importSources = {
	'file-upload': {
		label: __( 'File Upload', 'tablepress' ),
		instruction: __( 'Select files', 'tablepress' ),
	},
	url: {
		label: __( 'URL', 'tablepress' ),
		instruction: __( 'File URL', 'tablepress' ),
	},
	server: {
		label: __( 'File on server', 'tablepress' ),
		instruction: __( 'Server Path to file', 'tablepress' ),
	},
	'form-field': {
		label: __( 'Manual Input', 'tablepress' ),
		instruction: __( 'Import data', 'tablepress' ),
	},
};
if ( ! tp.import.showImportSourceUrl ) {
	delete importSources.url;
}
if ( ! tp.import.showImportSourceServer ) {
	delete importSources.server;
}

const importSourcesRadioOptions = Object.entries( importSources ).map( ( [ importSource, importSourceData ] ) => ( { value: importSource, label: importSourceData.label } ) );

// Number of tables.
const tablesCount = Object.keys( tp.import.tables ).length;

const tablesSelectOptions = Object.entries( tp.import.tables ).map( ( [ tableId, tableName ] ) => {
	if ( '' === tableName.trim() ) {
		tableName = __( '(no name)', 'tablepress' );
	}
	const optionText = sprintf( __( 'ID %1$s: %2$s', 'tablepress' ), tableId, tableName );
	return { value: tableId, label: optionText };
} );

// Custom component to conditionally disable its children, used for the ComboboxControl.
const ConditionalDisabled = ( { condition, children } ) => (
	condition ? ( <Disabled>{ children }</Disabled> ) : children
);

/**
 * Returns the "Import Screen" component's JSX markup.
 *
 * @return {Object} Import Screen component.
 */
const Screen = () => {
	const [ screenData, setScreenData ] = useState( {
		importSource: tp.import.importSource,
		importType: tp.import.importType,
		importFileUpload: [],
		importUrl: tp.import.importUrl,
		importServer: tp.import.importServer,
		importFormField: tp.import.importFormField,
		importExistingTable: tp.import.importExistingTable,
		validationHighlighting: false,
	} );

	/**
	 * Handles screen data state changes.
	 *
	 * @param {Object} updatedScreenData Data in the screen data state that should be updated.
	 */
	const updateScreenData = ( updatedScreenData ) => {
		setScreenData( ( currentScreenData ) => ( {
			...currentScreenData,
			validationHighlighting: false, // Reset with every UI state change.
			...updatedScreenData,
		} ) );
	};

	// References to DOM elements.
	const importServerInput = useRef( null );
	const fileUploadDropzone = useRef( null );

	// Update the validation highlighting (using APIs and DOM elements outside of the React components) when the state changes.
	useEffect( () => {
		document.getElementById( 'tablepress_import-import-form' ).classList.toggle( 'no-validation-highlighting', ! screenData.validationHighlighting );
		if ( ! screenData.validationHighlighting ) {
			importServerInput.current?.setCustomValidity( '' );
			// We need to use this dynamically generated ID by the ComboboxControl component. It does not (yet?) support a static ID or a ref.
			document.getElementById( 'components-form-token-input-combobox-control-1' )?.setCustomValidity( '' );
		}
	}, [ screenData.validationHighlighting ] );

	// Determine calculated state variables to avoid repeating calculations.
	const fileUploadMultipleFilesChosen = (
		'file-upload' === screenData.importSource
		&& (
			1 < screenData.importFileUpload.length
			|| (
				1 === screenData.importFileUpload.length
				&& screenData.importFileUpload[0].name.endsWith( '.zip' )
			)
		)
	);
	const appendReplaceDropdownDisabled = (
		0 === tablesCount
		|| 'add' === screenData.importType
		|| fileUploadMultipleFilesChosen
	);

	return (
		<table className="tablepress-postbox-table fixed">
			<tbody>
				<tr>
					<th className="column-1" scope="row">
						{ __( 'Import Source', 'tablepress' ) }:
					</th>
					<td className="column-2">
						<RadioControl
							name="import[source]"
							label={ __( 'Import Source', 'tablepress' ) }
							hideLabelFromVision={ true }
							selected={ screenData.importSource }
							onChange={ ( importSource ) => updateScreenData( { importSource } ) }
							options={ importSourcesRadioOptions }
						/>
					</td>
				</tr>
				<tr className="top-border bottom-border">
					<th className="column-1 top-align" scope="row">
						<label htmlFor={ `tables-import-${ screenData.importSource }` }>
							{ importSources[ screenData.importSource ].instruction }:
						</label>
					</th>
					<td className="column-2">
						{
							/*
							 * Always add the "File Upload" UI to the DOM, but hide it using `style="display: none;"` below.
							 * This ensures that the <input type="file"> field works, as that is "uncontrolled" in React, and setting its value (files) is not possible.
							 */
						}
						<div
							className="file-upload-area"
							style={ {
								display: ( 'file-upload' === screenData.importSource ) ? 'block' : 'none',
							} }
						>
							<input
								name="import_file_upload[]"
								id="tables-import-file-upload"
								type="file"
								multiple
								required={ 'file-upload' === screenData.importSource }
								onChange={ ( event ) => ( event.target.files && updateScreenData( { importFileUpload: event.target.files } ) ) }
								onDragEnter={ () => fileUploadDropzone.current.classList.add( 'dragover' ) }
								onDragLeave={ () => fileUploadDropzone.current.classList.remove( 'dragover' ) }
							/>
							<div
								ref={ fileUploadDropzone }
								id="tables-import-file-upload-dropzone"
								className="dropzone"
							>
								<span>
									{ 0 === screenData.importFileUpload.length && __( 'Click to select files, or drag them here.', 'tablepress' ) }
									{ 0 < screenData.importFileUpload.length && sprintf( _n( 'You have selected %1$d file:', 'You have selected %1$d files:', screenData.importFileUpload.length, 'tablepress' ), screenData.importFileUpload.length ) }
								</span>
								{
									[ ...screenData.importFileUpload ].map( ( file ) =>
										<span key={ file.name }>{ file.name }</span>
									)
								}
							</div>
						</div>
						{ tp.import.showImportSourceUrl && 'url' === screenData.importSource &&
							<TextControl
								__nextHasNoMarginBottom
								type="url"
								name="import[url]"
								id="tables-import-url"
								className="code"
								required={ true }
								value={ screenData.importUrl }
								onChange={ ( importUrl ) => updateScreenData( { importUrl } ) }
							/>
						}
						{ tp.import.showImportSourceServer && 'server' === screenData.importSource &&
							<TextControl
								__nextHasNoMarginBottom
								ref={ importServerInput }
								name="import[server]"
								id="tables-import-server"
								className="code"
								required={ true }
								value={ screenData.importServer }
								onChange={ ( importServer ) => updateScreenData( { importServer } ) }
							/>
						}
						{ 'form-field' === screenData.importSource &&
							<TextareaControl
								__nextHasNoMarginBottom
								name="import[form-field]"
								id="tables-import-form-field"
								rows="15"
								cols="40"
								className="code"
								required={ true }
								value={ screenData.importFormField }
								onChange={ ( importFormField ) => updateScreenData( { importFormField } ) }
							/>
						}
						{ 'form-field' !== screenData.importSource &&
							<HStack
								alignment="left"
							>
								<Icon icon={ info } />
								<span>
									{ __( 'You can also import multiple tables by placing them in a ZIP file.', 'tablepress' ) }
								</span>
							</HStack>
						}
					</td>
				</tr>
				<tr className="top-border">
					<th className="column-1" scope="row">
						{ __( 'Add, Replace, or Append?', 'tablepress' ) }:
					</th>
					<td className="column-2">
						<RadioControl
							name="import[type]"
							label={ __( 'Import Type', 'tablepress' ) }
							hideLabelFromVision={ true }
							selected={ 0 === tablesCount ? 'add' : screenData.importType } // Always select "Add" if there are no tables.
							onChange={ ( importType ) => updateScreenData( { importType } ) }
							options={ [
								{ value: 'add', label: __( 'Add as new table', 'tablepress' ) },
								{ value: 'replace', label: __( 'Replace existing table', 'tablepress' ), disabled: 0 === tablesCount },
								{ value: 'append', label: __( 'Append rows to existing table', 'tablepress' ), disabled: 0 === tablesCount },
							] }
						/>
					</td>
				</tr>
				<tr className="top-border bottom-border">
					<th className="column-1 top-align" scope="row">
						<label htmlFor="tables-import-existing-table">{ __( 'Table to replace or append to', 'tablepress' ) }:</label>
					</th>
					<td className="column-2">
						<ConditionalDisabled
							condition={ appendReplaceDropdownDisabled }
						>
							<ComboboxControl
								__nextHasNoMarginBottom
								id="tables-import-existing-table"
								label={ __( 'Table to replace or append to', 'tablepress' ) }
								hideLabelFromVision={ true }
								placeholder={ __( '— Select or type —', 'tablepress' ) }
								value={ screenData.importExistingTable }
								options={ tablesSelectOptions }
								onChange={ ( importExistingTable ) => updateScreenData( { importExistingTable } ) }
							/>
						</ConditionalDisabled>
					</td>
				</tr>
				<tr className="top-border">
					<td className="column-1"></td>
					<td className="column-2">
						<input
							// Send the legacy import flag to the server, so that it can handle the import accordingly.
							type="hidden"
							name="import[legacy_import]"
							value={ tp.import.legacyImport }
						/>
						<input
							// Send the Table to be replaced/appended to the server, if a table was selected. The ComboboxControl is not an actual form element with a name.
							type="hidden"
							name="import[existing_table]"
							value={ screenData.importExistingTable ?? '' }
						/>
						<Button
							variant="primary"
							type="submit"
							text={ _x( 'Import', 'button', 'tablepress' ) }
							onClick={ () => {
								// Show validation :invalid CSS pseudo-selector highlighting.
								updateScreenData( { validationHighlighting: true } );

								// When importing from the server, the value must have been changed from the default (normally ABSPATH).
								if ( 'server' === screenData.importSource && tp.import.importServer === screenData.importServer ) {
									importServerInput.current.setCustomValidity( __( 'You must specify a path to a file on the server.', 'tablepress' ) );
								}

								// If the table selection dropdown for replace or append is enabled, a table must be selected.
								if ( ! appendReplaceDropdownDisabled && ! screenData.importExistingTable ) {
									// We need to use this dynamically generated ID by the ComboboxControl component. It does not (yet?) support a static ID or a ref.
									document.getElementById( 'components-form-token-input-combobox-control-1' )?.setCustomValidity( __( 'You must select a table.', 'tablepress' ) );
								}
							} }
						/>
					</td>
				</tr>
			</tbody>
		</table>
	);
};

export default Screen;