Importador XML

<?php
/**
 * Plugin Name: Importador XML Avanzado
 * Description: Permite subir un archivo XML desde el panel de administración y genera entradas evitando duplicados.
 * Version: 1.1
 * Author: Juan Pablo
 */

if ( ! defined( 'ABSPATH' ) ) exit; // seguridad

// Menú en administración
add_action('admin_menu', 'importador_xml_menu');
function importador_xml_menu() {
    add_menu_page(
        'Importador XML',
        'Importador XML',
        'manage_options',
        'importador-xml',
        'importador_xml_page',
        'dashicons-database-import'
    );
}

// Página del plugin
function importador_xml_page() {
    ?>
    <div class="wrap">
        <h1>Importador de Entradas desde XML</h1>

        <h2>Subir XML</h2>
        <form method="post" enctype="multipart/form-data">
            <?php wp_nonce_field('importador_xml_nonce','importador_xml_nonce_field'); ?>
            <input type="file" name="archivo_xml" accept=".xml" required>
            <input type="submit" name="subir_xml" class="button button-secondary" value="Subir XML">
        </form>

        <h2>Importar entradas</h2>
        <form method="post">
            <?php wp_nonce_field('importador_xml_nonce','importador_xml_nonce_field'); ?>
            <input type="submit" name="importar_xml" class="button button-primary" value="Importar desde XML">
        </form>
    </div>
    <?php

    // Procesar subida
    if (isset($_POST['subir_xml']) && check_admin_referer('importador_xml_nonce','importador_xml_nonce_field')) {
        importador_xml_subir();
    }

    // Procesar importación
    if (isset($_POST['importar_xml']) && check_admin_referer('importador_xml_nonce','importador_xml_nonce_field')) {
        importador_xml_procesar();
    }
}

// Subir archivo XML a /uploads/importador-xml/
function importador_xml_subir() {
    if (!empty($_FILES['archivo_xml']['tmp_name'])) {
        $upload_dir = wp_upload_dir();
        $destino_dir = $upload_dir['basedir'] . '/importador-xml/';

        if (!file_exists($destino_dir)) {
            wp_mkdir_p($destino_dir);
        }

        $destino = $destino_dir . 'datos.xml';

        if (move_uploaded_file($_FILES['archivo_xml']['tmp_name'], $destino)) {
            echo "<div class='updated'><p>Archivo subido correctamente: " . esc_html($destino) . "</p></div>";
        } else {
            echo "<div class='error'><p>Error al subir el archivo.</p></div>";
        }
    }
}

// Procesar archivo XML y crear posts
function importador_xml_procesar() {
    $upload_dir = wp_upload_dir();
    $ruta = $upload_dir['basedir'] . '/importador-xml/datos.xml';

    if (!file_exists($ruta)) {
        echo "<div class='error'><p>No se encontró ningún archivo XML en: $ruta</p></div>";
        return;
    }

    $xml = simplexml_load_file($ruta);
    if (!$xml) {
        echo "<div class='error'><p>Error al leer el archivo XML.</p></div>";
        return;
    }

    $contador = 0;
    foreach ($xml->item as $item) {
        $titulo = sanitize_text_field((string) $item->title);
        $contenido = wp_kses_post((string) $item->body);

        // Comprobar si ya existe un post con este título
        $existe = get_page_by_title($titulo, OBJECT, 'post');
        if (!$existe) {
           
            $nuevo_post = array(
                'post_title'   => $titulo,
                'post_content' => $contenido,
                'post_status'  => 'publish',
                'post_author'  => get_current_user_id(),
                'post_type'    => 'post'
            );

            wp_insert_post($nuevo_post);
            $contador++;
        }
    }

    echo "<div class='updated'><p>Se importaron $contador nuevas entradas desde el XML.</p></div>";
}

Ejemplo plugin consulta datos externos

Vamos a crear un plugin que consulte datos externos de una API y los muestre vía shortcode

1.- Crear el shortcode


<?php
/*
Plugin Name: Mostrar datos de una API externa
Description: Consulta los datos de una API y los muestra donde yo ponga el shortcode
Version: 1.0
Author: Jotapeich
*/

function mostrar_datos_func( $atts ) {
  
    return "Aquí se mostrarán los datos";
}
add_shortcode( 'datos_api', 'mostrar_datos_func' );

2.- Obtener los datos de una api externa

<?php
/*
Plugin Name: Mostrar datos de una API externa
Description: Consulta los datos de una API y los muestra donde yo ponga el shortcode
Version: 1.0
Author: Jotapeich
*/
function consultar_api( $cantidad = 5 ) {
    $url = "https://jsonplaceholder.typicode.com/posts"; // API de ejemplo

    $response = wp_remote_get( $url );
    if ( is_wp_error( $response ) ) {
        return "<p>Error al consultar la API.</p>";
    }

    $body = wp_remote_retrieve_body( $response );
    $data = json_decode( $body, true );

    if ( ! is_array( $data ) ) {
        return "<p>Respuesta inválida de la API.</p>";
    } 
    
}
function mostrar_datos_func( $atts ) {
    consultar_api();
    return "Aquí se mostrarán los datos";
}
add_shortcode( 'datos_api', 'mostrar_datos_func' );

3.- Formatear los datos que me llegan:

<?php
/*
Plugin Name: Mostrar datos de una API externa
Description: Consulta los datos de una API y los muestra donde yo ponga el shortcode
Version: 1.0
Author: Jotapeich
*/
function consultar_api( $cantidad = 5 ) {
    $url = "https://jsonplaceholder.typicode.com/posts"; // API de ejemplo

    $response = wp_remote_get( $url );
    if ( is_wp_error( $response ) ) {
        return "<p>Error al consultar la API.</p>";
    }

    $body = wp_remote_retrieve_body( $response );
    $data = json_decode( $body, true );

    if ( ! is_array( $data ) ) {
        return "<p>Respuesta inválida de la API.</p>";
    } 
     $data = array_slice( $data, 0, intval($cantidad) );

    $output  = '<table style="border-collapse: collapse; width:100%; border:1px solid #ccc;">';
    $output .= '<thead><tr style="background:#f4f4f4;"><th style="padding:8px;border:1px solid #ccc;">ID</th><th style="padding:8px;border:1px solid #ccc;">Título</th><th style="padding:8px;border:1px solid #ccc;">Body</th></tr></thead><tbody>';

    foreach ( $data as $item ) {
        $output .= '<tr>';
        $output .= '<td style="padding:8px;border:1px solid #ccc;">' . esc_html($item['id']) . '</td>';
        $output .= '<td style="padding:8px;border:1px solid #ccc;">' . esc_html($item['title']) . '</td>';
        $output .= '<td style="padding:8px;border:1px solid #ccc;">' . nl2br(esc_html($item['body'])) . '</td>';
        $output .= '</tr>';
    }

    $output .= '</tbody></table>';

    return $output;
}
function mostrar_datos_func( $atts ) {
    
    return consultar_api();
}
add_shortcode( 'datos_api', 'mostrar_datos_func' );

4.- Le voy a poner un parámetro al shortcode

function mostrar_datos_func( $atts ) {
     $atts = shortcode_atts( array(
        'cantidad' => 5,
    ), $atts, 'mis_datos' );
    return consultar_api($atts['cantidad']);
}

5.- Vamos a añadir opciones

<?php
/*
Plugin Name: Mostrar datos de una API externa
Description: Consulta los datos de una API y los muestra donde yo ponga el shortcode
Version: 1.0
Author: Jotapeich
*/
function consultar_api( $cantidad = 5 ) {
    $url = "https://jsonplaceholder.typicode.com/posts"; // API de ejemplo

    $response = wp_remote_get( $url );
    if ( is_wp_error( $response ) ) {
        return "<p>Error al consultar la API.</p>";
    }

    $body = wp_remote_retrieve_body( $response );
    $data = json_decode( $body, true );

    if ( ! is_array( $data ) ) {
        return "<p>Respuesta inválida de la API.</p>";
    } 
     $data = array_slice( $data, 0, intval($cantidad) );

    $output  = '<table style="border-collapse: collapse; width:100%; border:1px solid #ccc;">';
    $output .= '<thead><tr style="background:#f4f4f4;"><th style="padding:8px;border:1px solid #ccc;">ID</th><th style="padding:8px;border:1px solid #ccc;">Título</th><th style="padding:8px;border:1px solid #ccc;">Body</th></tr></thead><tbody>';

    foreach ( $data as $item ) {
        $output .= '<tr>';
        $output .= '<td style="padding:8px;border:1px solid #ccc;">' . esc_html($item['id']) . '</td>';
        $output .= '<td style="padding:8px;border:1px solid #ccc;">' . esc_html($item['title']) . '</td>';
        $output .= '<td style="padding:8px;border:1px solid #ccc;">' . nl2br(esc_html($item['body'])) . '</td>';
        $output .= '</tr>';
    }

    $output .= '</tbody></table>';

    return $output;
}
function mostrar_datos_func( $atts ) {

    $default_cantidad = get_option( 'asp_cantidad_default', 5 );
     $atts = shortcode_atts( array(
        'cantidad' => $default_cantidad,
    ), $atts, 'mis_datos' );
    return consultar_api($atts['cantidad']);
}
add_shortcode( 'datos_api', 'mostrar_datos_func' );


// ===========================
// 3. Menú en administración
// ===========================
function asp_add_admin_menu() {
    add_menu_page(
        'Configuración API', // Título de la página
        'Configuración API',              // Texto en el menú
        'manage_options',             // Capacidad requerida
        'asp-config',                 // Slug del menú
        'asp_options_page_html',      // Callback
        'dashicons-admin-generic',    // Icono
        20                            // Posición
    );
}
add_action( 'admin_menu', 'asp_add_admin_menu' );

// ===========================
// 4. Registrar ajustes
// ===========================
function asp_settings_init() {
    register_setting( 'asp_options_group', 'asp_cantidad_default', array(
        'type' => 'integer',
        'default' => 5,
    ) );

    add_settings_section(
        'asp_section',
        'Configuración del shortcode',
        '__return_false',
        'asp_options_group'
    );

    add_settings_field(
        'asp_cantidad_default',
        'Cantidad por defecto',
        'asp_field_cantidad_render',
        'asp_options_group',
        'asp_section'
    );
}
add_action( 'admin_init', 'asp_settings_init' );

// Campo de formulario
function asp_field_cantidad_render() {
    $valor = get_option( 'asp_cantidad_default', 5 );
    echo '<input type="number" name="asp_cantidad_default" value="' . esc_attr( $valor ) . '" min="1" max="50">';
}

// ===========================
// 5. Página HTML en admin
// ===========================
function asp_options_page_html() {
    if ( ! current_user_can( 'manage_options' ) ) {
        return;
    }
    ?>
    <div class="wrap">
        <h1>Configuración de los datos de la api externa</h1>
        <form action="options.php" method="post">
            <?php
            settings_fields( 'asp_options_group' );
            do_settings_sections( 'asp_options_group' );
            submit_button();
            ?>
        </form>
    </div>
    <?php
}

Ejemplo opciones en WordPress

<?php
/**
 * Plugin Name: Contar palabras
 * Description: Cuenta las palabras que tiene una entrada del blog y lo pone al final
 * Version: 1.1
 * Author: Jotapé
 */


// Esta función se encarga de contar las palabras y añadirlas
function contar_palabras($content){
    $ppm=get_option( 'palabras_minuto_default', 200 );
    // usamos str_word_count para contar las palabras: se puede mejorar para no contar html o imágenes o etc...
    $num_palabras=str_word_count($content);
    // Calculo el tiempo de lectura
    $tiempo=round($num_palabras/$ppm,1);
    // Creo el texto a añadir. Como son comillas dobles puedo poner la variable dentro
    $texto="Número de palabras: <b>$num_palabras</b>. Tiempo estimado de lectura: <b>$tiempo minutos</b>";

    // Devuelvo el contenido con el texto anterior. 
    return "<p>$texto</p>$content";
}

// Añado el filtro para modificar el contenido
add_filter("the_content","contar_palabras");

function ppm_add_admin_menu() {
    add_menu_page(
        'Palabras por minuto', // Título de la página
        'Config PPM',              // Texto en el menú
        'manage_options',             // Capacidad requerida
        'ppm-config',                 // Slug del menú
        'ppm_options_page_html',      // Callback
        'dashicons-admin-generic',    // Icono
        20                            // Posición
    );
}
add_action( 'admin_menu', 'ppm_add_admin_menu' );

function ppm_settings_init() {
    register_setting( 'options_contar_palabras', 'palabras_minuto_default', array(
        'type' => 'integer',        
    'sanitize_callback' => 'absint',
        'default' => 200,
    ) );

    add_settings_section(
        'contar_palabras_section',
        'Configuración de las palabras por minuto',
        function() { echo '<p>Configura el valor de las palabras por minuto para estimar el tiempo de lectura.</p>'; },
        'options_contar_palabras'
    );

    add_settings_field(
        'palabras_minuto_default',
        'Palabras por minuto por defecto',
        'asp_field_palabras_render',
        'options_contar_palabras',
        'contar_palabras_section'
    );
}
add_action( 'admin_init', 'ppm_settings_init' );

function asp_field_palabras_render() {
    $valor = get_option( 'palabras_minuto_default', 200 );
    echo '<input type="number" name="palabras_minuto_default" value="' . esc_attr( $valor ) . '" min="50" max="500">';
}

function ppm_options_page_html() {
    if ( ! current_user_can( 'manage_options' ) ) {
        return;
    }
    ?>
    <div class="wrap">
        <h1>Configuración de las palabras por minuto por defecto</h1>
        <form action="options.php" method="post">
            <?php
            settings_fields( 'options_contar_palabras' );
            do_settings_sections( 'options_contar_palabras' );
            submit_button();
            ?>
        </form>
    </div>
    <?php
}