0

我想将 PDF 导出集成到我的 SONATA ADMIN 应用程序中。为此,我安装了 KnpSnappy Bundle 和 SonataExporterBundle。我遵循了在谷歌上找到的旧教程,但最终它不起作用。

配置.yml:

knp_snappy:
pdf:
    enabled:    true
    binary:     /usr/local/bin/wkhtmltopdf 
    options:    []
image:
    enabled:    true
    binary:     /usr/local/bin/wkhtmltoimage 
    options:    []  
temporary_folder: %kernel.cache_dir%/snappy    

服务.yml:

    sonata.admin.exporter:
        class: AppBundle\Export\Exporter
        calls:
            - ["setKnpSnappyPdf", ["@knp_snappy.pdf"]]
            - ["setTemplateEngine", ["@templating"] ]

在我的 ModelAdmin 中,我添加了它:

    public function getExportFormats() {
    return array_merge(parent::getExportFormats(), array('pdf'));
}

我创建了 AppBundle/Export/Exporter.php:

    namespace AppBundle\Export;

use Exporter\Source\SourceIteratorInterface;
use Symfony\Component\HttpFoundation\Response;
use Sonata\AdminBundle\Export\Exporter as BaseExporter;

class Exporter extends BaseExporter
{
  protected $knpSnappyPdf;
  protected $templateEngine;

  public function getResponse($format, $filename, SourceIteratorInterface $source)
  {
    if ('pdf' != $format) {
      return parent::getResponse($format, $filename, $source);
    }

    $html = $this->templateEngine->renderView('AppBundle:Export:pdf.html.twig', array(
      'source' => $source
    ));
    $content = $this->knpSnappyPdf->getOutputFromHtml($html);

    return new Response($content, 200, array(
      'Content-Type' => 'application/pdf',
      'Content-Disposition' => sprintf('attachment; filename=%s', $filename)
    ));
  }

  public function setKnpSnappyPdf($service)
  {
    $this->knpSnappyPdf = $service;
  }

  public function setTemplateEngine($service)
  {
    $this->templateEngine = $service;
  }
}

错误:

RuntimeException:
Invalid "pdf" format, supported formats are : "csv, json, xls, xml"

  at vendor/sonata-project/exporter/src/Exporter.php:52
  at Exporter\Exporter->getResponse('pdf', 'export_model_2018_04_20_15_44_05.pdf', object(DoctrineORMQuerySourceIterator))
     (vendor/sonata-project/admin-bundle/src/Controller/CRUDController.php:952)
  at Sonata\AdminBundle\Controller\CRUDController->exportAction(object(Request))
     (vendor/symfony/symfony/src/Symfony/Component/HttpKernel/HttpKernel.php:151)
  at Symfony\Component\HttpKernel\HttpKernel->handleRaw(object(Request), 1)
     (vendor/symfony/symfony/src/Symfony/Component/HttpKernel/HttpKernel.php:68)
  at Symfony\Component\HttpKernel\HttpKernel->handle(object(Request), 1, true)
     (vendor/symfony/symfony/src/Symfony/Component/HttpKernel/Kernel.php:202)
  at Symfony\Component\HttpKernel\Kernel->handle(object(Request))
     (web/app_dev.php:31)
  at require('/var/www/html/acianovintra/web/app_dev.php')
     (vendor/symfony/symfony/src/Symfony/Bundle/WebServerBundle/Resources/router.php:42)

你能告诉我我做错了什么吗?

4

1 回答 1

0

在一个项目中,我必须以 docx 格式导出一些数据。

为此,我编写了一个自定义编写器:

<?php

namespace App\MyBundle\Utils\Exporter\Writer;

use Exporter\Writer\TypedWriterInterface;

class DocxWriter implements TypedWriterInterface
{
    /**
     * {@inheritdoc}
     */
    final public function getDefaultMimeType()
    {
        return 'application/vnd.openxmlformats-officedocument.wordprocessingml.document';
    }

    /**
     * {@inheritdoc}
     */
    final public function getFormat()
    {
        return 'docx';
    }

    //...
}

然后我必须为作者创建一个服务并添加一个标签:

// service.xml
 <?xml version="1.0" ?>

<container xmlns="http://symfony.com/schema/dic/services"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://symfony.com/schema/dic/services http://symfony.com/schema/dic/services/services-1.0.xsd">
    <services>
        <service id="app.exporter.writer.docx" class="App\MyBundle\Utils\Exporter\Writer\DocxWriter">
            <argument>php://output</argument>
            <tag name="sonata.exporter.writer"/>
        </service>
    </services>
</container>

ExporterCompilerPassexporter 捆绑包中,sonata 获取所有标记为的服务sonata.exporter.writer并将它们传递给addWriterExporter

要将这个新作家集成到奏鸣曲管理中,你必须sonata.admin.exporter像你已经做过的那样覆盖,除了你将调用这个新作家:

<?php
namespace App\MyBundle\Export;

use Exporter\Source\SourceIteratorInterface;
use Sonata\AdminBundle\Export\Exporter as BaseExporter;
use App\MyBundle\Utils\Exporter\Writer\DocxWriter;
use Symfony\Component\HttpFoundation\StreamedResponse;

class Exporter extends BaseExporter
{
    public function getResponse($format, $filename, SourceIteratorInterface $source)
    {
        switch ($format) {
            case 'docx':
                $writer = new DocxWriter('php://output');
                $contentType = 'application/vnd.openxmlformats-officedocument.wordprocessingml.document';
                break;
            default:
                return parent::getResponse($format, $filename, $source);
        }

        $callback = function () use ($source, $writer) {
            $handler = \Exporter\Handler::create($source, $writer);
            $handler->export();
        };

        return new StreamedResponse($callback, 200, array(
            'Content-Type' => $contentType,
            'Content-Disposition' => sprintf('attachment; filename="%s"', $filename),
        ));
    }
}

有了这个,您应该能够在您的管理员中添加您的 pdf 格式来覆盖该getExportFormats方法。

希望这有帮助

于 2018-04-24T17:49:53.323 回答