steps = array( 'upload' => array( 'name' => __( 'Upload CSV file', 'woocommerce' ), 'view' => array( $this, 'upload_form' ), 'handler' => array( $this, 'upload_form_handler' ), ), 'mapping' => array( 'name' => __( 'Column mapping', 'woocommerce' ), 'view' => array( $this, 'mapping_form' ), 'handler' => '', ), 'import' => array( 'name' => __( 'Import', 'woocommerce' ), 'view' => array( $this, 'import' ), 'handler' => '', ), 'done' => array( 'name' => __( 'Done!', 'woocommerce' ), 'view' => array( $this, 'done' ), 'handler' => '', ), ); $this->step = isset( $_REQUEST['step'] ) ? sanitize_key( $_REQUEST['step'] ) : current( array_keys( $this->steps ) ); $this->file = isset( $_REQUEST['file'] ) ? wc_clean( $_REQUEST['file'] ) : ''; } /** * Get the URL for the next step's screen. * @param string step slug (default: current step) * @return string URL for next step if a next step exists. * Admin URL if it's the last step. * Empty string on failure. */ public function get_next_step_link( $step = '' ) { if ( ! $step ) { $step = $this->step; } $keys = array_keys( $this->steps ); if ( end( $keys ) === $step ) { return admin_url(); } $step_index = array_search( $step, $keys ); if ( false === $step_index ) { return ''; } $params = array( 'step' => $keys[ $step_index + 1 ], 'file' => $this->file, 'delimiter' => $this->delimiter, '_wpnonce' => wp_create_nonce( 'woocommerce-csv-importer' ), // wp_nonce_url() escapes & to & breaking redirects. ); return add_query_arg( $params ); } /** * Output header view. */ protected function output_header() { include( dirname( __FILE__ ) . '/views/html-csv-import-header.php' ); } /** * Output steps view. */ protected function output_steps() { include( dirname( __FILE__ ) . '/views/html-csv-import-steps.php' ); } /** * Output footer view. */ protected function output_footer() { include( dirname( __FILE__ ) . '/views/html-csv-import-footer.php' ); } /** * Get importer instance. * * @param string $file File to import. * @param array $args Importer arguments. * @return WC_Product_CSV_Importer */ protected function get_importer( $file, $args = array() ) { $importer_class = apply_filters( 'woocommerce_product_csv_importer_class', 'WC_Product_CSV_Importer' ); return new $importer_class( $file, $args ); } /** * Add error message. */ protected function add_error( $error ) { $this->errors[] = $error; } /** * Add error message. */ protected function output_errors() { if ( $this->errors ) { foreach ( $this->errors as $error ) { echo '
' . esc_html( $error ) . '
'; } } } /** * Dispatch current step and show correct view. */ public function dispatch() { if ( ! empty( $_POST['save_step'] ) && ! empty( $this->steps[ $this->step ]['handler'] ) ) { call_user_func( $this->steps[ $this->step ]['handler'], $this ); } $this->output_header(); $this->output_steps(); $this->output_errors(); call_user_func( $this->steps[ $this->step ]['view'], $this ); $this->output_footer(); } /** * Output information about the uploading process. */ protected function upload_form() { $bytes = apply_filters( 'import_upload_size_limit', wp_max_upload_size() ); $size = size_format( $bytes ); $upload_dir = wp_upload_dir(); include( dirname( __FILE__ ) . '/views/html-product-csv-import-form.php' ); } /** * Handle the upload form and store options. */ public function upload_form_handler() { check_admin_referer( 'woocommerce-csv-importer' ); $file = $this->handle_upload(); if ( is_wp_error( $file ) ) { $this->add_error( $file->get_error_message() ); return; } else { $this->file = $file; } wp_redirect( esc_url_raw( $this->get_next_step_link() ) ); exit; } /** * Handles the CSV upload and initial parsing of the file to prepare for. * displaying author import options. * * @return string|WP_Error */ public function handle_upload() { if ( empty( $_POST['file_url'] ) ) { if ( ! isset( $_FILES['import'] ) ) { return new WP_Error( __( 'File is empty. Please upload something more substantial. This error could also be caused by uploads being disabled in your php.ini or by post_max_size being defined as smaller than upload_max_filesize in php.ini.', 'woocommerce' ) ); } $overrides = array( 'test_form' => false, 'test_type' => false ); $_FILES['import']['name'] .= '.txt'; $upload = wp_handle_upload( $_FILES['import'], $overrides ); if ( isset( $upload['error'] ) ) { return new WP_Error( $upload['error'] ); } // Construct the object array $object = array( 'post_title' => basename( $upload['file'] ), 'post_content' => $upload['url'], 'post_mime_type' => $upload['type'], 'guid' => $upload['url'], 'context' => 'import', 'post_status' => 'private', ); // Save the data $id = wp_insert_attachment( $object, $upload['file'] ); /* * Schedule a cleanup for one day from now in case of failed * import or missing wp_import_cleanup() call. */ wp_schedule_single_event( time() + DAY_IN_SECONDS, 'importer_scheduled_cleanup', array( $id ) ); return $upload['file']; } elseif ( file_exists( ABSPATH . $_POST['file_url'] ) ) { return ABSPATH . $_POST['file_url']; } return new WP_Error( __( 'Please upload or provide the link to a valid CSV file.', 'woocommerce' ) ); } /** * Mapping step @todo */ protected function mapping_form() { $importer = $this->get_importer( $this->file, array( 'lines' => 1 ) ); $headers = $importer->get_raw_keys(); $sample = current( $importer->get_raw_data() ); if ( empty( $sample ) ) { $this->add_error( __( 'The file is empty, please try again with a new file.', 'woocommerce' ) ); return; } // Check if all fields matches. if ( 0 === count( array_diff( $headers, $this->get_default_fields() ) ) ) { wp_redirect( esc_url_raw( $this->get_next_step_link() ) ); exit; } include_once( dirname( __FILE__ ) . '/views/html-csv-import-mapping.php' ); } /** * Import the file if it exists and is valid. */ public function import() { if ( ! is_file( $this->file ) ) { $this->add_error( __( 'The file does not exist, please try again.', 'woocommerce' ) ); return; } $mapping = array(); if ( ! empty( $_POST['map_to'] ) ) { $mapping = wp_unslash( $_POST['map_to'] ); } include_once( dirname( __FILE__ ) . '/views/html-csv-import-progress.php' ); wp_localize_script( 'wc-product-import', 'wc_product_import_params', array( 'import_nonce' => wp_create_nonce( 'wc-product-import' ), 'mapping' => $mapping, 'file' => $this->file, ) ); wp_enqueue_script( 'wc-product-import' ); } /** * Done step. * @todo Make this better. */ protected function done() { $imported = isset( $_GET['imported'] ) ? absint( $_GET['imported'] ) : 0; $failed = isset( $_GET['failed'] ) ? absint( $_GET['failed'] ) : 0; $results = sprintf( /* translators: %d: products count */ _n( 'Imported %s product.', 'Imported %s products.', $imported, 'woocommerce' ), '' . number_format_i18n( $imported ) . '' ); // @todo create a view to display errors or log with WC_Logger. if ( 0 < $failed ) { $results .= ' ' . sprintf( /* translators: %d: products count */ _n( 'Failed %s product.', 'Failed %s products.', $failed, 'woocommerce' ), '' . number_format_i18n( $failed ) . '' ); } // Show result. echo ''; /* translators: %d: import results */ printf( __( 'Import complete: %s', 'woocommerce' ), $results ); echo '