En este post te quiero compartir un código de C++ para listar y cancelar trabajos de impresión de la cola de impresión de una impresora en Windows.

Con el código C++ vamos a usar la API de Windows (win32) con EnumJobs para obtener los trabajos de impresión de una impresora determinada, y después, opcionalmente, podremos cancelar ese trabajo con SetJob ya sea basándonos en el índice, usuario o cualquier otro dato del trabajo.

Abriendo impresora y listando trabajos

Comenzamos obteniendo un manejador de la impresora a partir de su nombre:

HANDLE manejadorImpresora = NULL;
PRINTER_DEFAULTSW pd = {nullptr, nullptr, PRINTER_ACCESS_USE};
if (!OpenPrinterW(const_cast<LPWSTR>(nombreImpresora.c_str()), &manejadorImpresora, &pd))
{
    std::wcerr << L"Error abriendo impresora " << nombreImpresora << std::endl;
    return;
}

Luego enumeramos los trabajos de impresión:

DWORD trabajos = 0, cantidadBytes = 0;
EnumJobsW(manejadorImpresora, 0, 0xFFFFFFFF, 1, nullptr, 0, &cantidadBytes, &trabajos);
if (cantidadBytes <= 0)
{

    std::wcout << L"Cola vacia" << std::endl;
    return;
}

Si hay trabajos, los recorremos:

auto trabajoDeImpresion = (JOB_INFO_1W *)malloc(cantidadBytes);
if (trabajoDeImpresion && EnumJobsW(manejadorImpresora, 0, 0xFFFFFFFF, 1, (LPBYTE)trabajoDeImpresion, cantidadBytes, &cantidadBytes, &trabajos))
{
    for (DWORD indice = 0; indice < trabajos; ++indice)
    {
        std::wcout << L"Trabajo " << trabajoDeImpresion[indice].JobId << L":" << std::endl;
        std::wcout << L"  Documento: "
                    << (trabajoDeImpresion[indice].pDocument ? trabajoDeImpresion[indice].pDocument : L"(Sin nombre)")
                    << std::endl;
        std::wcout << L"  Propietario: "
                    << (trabajoDeImpresion[indice].pUserName ? trabajoDeImpresion[indice].pUserName : L"(Desconocido)")
                    << std::endl;
        std::wcout << L"  Páginas impresas: " << trabajoDeImpresion[indice].PagesPrinted
                    << L" de " << trabajoDeImpresion[indice].TotalPages << std::endl;
        std::wcout << L"  Estado: " << trabajoDeImpresion[indice].Status << std::endl;
        // Y la siguiente línea lo cancela con JOB_CONTROL_CANCEL
        // Puedes tener un if y comprobar el índice, pUserName, etcétera
        SetJob(manejadorImpresora, trabajoDeImpresion[indice].JobId, 0, nullptr, JOB_CONTROL_CANCEL);
    }
}

Cancelar trabajo de impresión

La línea que cancela el trabajo de impresión es SetJob con la opción JOB_CONTROL_CANCEL y ya está dentro del ciclo que te mostré en el apartado anterior:

SetJob(manejadorImpresora, trabajoDeImpresion[indice].JobId, 0, nullptr, JOB_CONTROL_CANCEL);

De hecho no es obligatorio que imprimas los datos de los trabajos pendientes. También puedes hacer un if revisando el id, usuario, etcétera para saber si deberías cancelar el trabajo de impresión.

Código completo

El código completo queda como se ve a continuación. He encerrado todo el comportamiento en una función que puedes invocar desde el main. Dicha función recibe el nombre de la impresora tal y como aparece en el panel de control:

#include <windows.h>
#include <iostream>
#include <string>

void imprimirTrabajosYCancelar(const std::wstring &nombreImpresora)
{
    HANDLE manejadorImpresora = NULL;
    PRINTER_DEFAULTSW pd = {nullptr, nullptr, PRINTER_ACCESS_USE};
    if (!OpenPrinterW(const_cast<LPWSTR>(nombreImpresora.c_str()), &manejadorImpresora, &pd))
    {
        std::wcerr << L"Error abriendo impresora " << nombreImpresora << std::endl;
        return;
    }
    DWORD trabajos = 0, cantidadBytes = 0;
    EnumJobsW(manejadorImpresora, 0, 0xFFFFFFFF, 1, nullptr, 0, &cantidadBytes, &trabajos);
    if (cantidadBytes <= 0)
    {

        std::wcout << L"Cola vacia" << std::endl;
        return;
    }

    auto trabajoDeImpresion = (JOB_INFO_1W *)malloc(cantidadBytes);
    if (trabajoDeImpresion && EnumJobsW(manejadorImpresora, 0, 0xFFFFFFFF, 1, (LPBYTE)trabajoDeImpresion, cantidadBytes, &cantidadBytes, &trabajos))
    {
        for (DWORD indice = 0; indice < trabajos; ++indice)
        {
            std::wcout << L"Trabajo " << trabajoDeImpresion[indice].JobId << L":" << std::endl;
            std::wcout << L"  Documento: "
                       << (trabajoDeImpresion[indice].pDocument ? trabajoDeImpresion[indice].pDocument : L"(Sin nombre)")
                       << std::endl;
            std::wcout << L"  Propietario: "
                       << (trabajoDeImpresion[indice].pUserName ? trabajoDeImpresion[indice].pUserName : L"(Desconocido)")
                       << std::endl;
            std::wcout << L"  Páginas impresas: " << trabajoDeImpresion[indice].PagesPrinted
                       << L" de " << trabajoDeImpresion[indice].TotalPages << std::endl;
            std::wcout << L"  Estado: " << trabajoDeImpresion[indice].Status << std::endl;
            // Y la siguiente línea lo cancela con JOB_CONTROL_CANCEL
            // Puedes tener un if y comprobar el índice, pUserName, etcétera
            SetJob(manejadorImpresora, trabajoDeImpresion[indice].JobId, 0, nullptr, JOB_CONTROL_CANCEL);
        }
    }
    free(trabajoDeImpresion);
    ClosePrinter(manejadorImpresora);
}

int main()
{
    // Nombre de tu impresora tal y como aparece en el panel de control
    std::wstring nombreImpresora = L"POS58 Printer";
    imprimirTrabajosYCancelar(nombreImpresora);
    return 0;
}

Para compilarlo hay que pasar el parámetro -lwinspool así:

g++ main.cpp -lwinspool -o cola_impresion.exe

Si el post ha sido de tu agrado te invito a que me sigas para saber cuando haya escrito un nuevo post, haya actualizado algún sistema o publicado un nuevo software. Facebook | X | Instagram | Telegram | También estoy a tus órdenes para cualquier contratación en mi página de contacto