Download de PDF via POST com HttpClient no Angular
Marcelo Ribas Vismari -
Incluir um link (<a>) em seu HTML para iniciar um download é uma prática comum de implementar. No entanto, em outro cenário no qual já precisei trabalhar, o frontend precisava enviar algumas informações no body e no header de uma requisição POST. Essas informações eram processadas pelo backend, que então retornava um arquivo PDF. Este cenário não é tão direto quanto o primeiro exemplo. Portanto, gostaria de compartilhar como podemos exibir e disponibilizar o download de um arquivo PDF através de uma solicitação POST usando o HttpClient do Angular.
Estruturando o cenário
O cenário é bem simples, vou criar um simples backend que será executado em Node.js e depois um novo projeto Angular.
Criando o backend para fornecer o PDF
Para criar o projeto utilizei o script abaixo:
# Crio um diretório:
mkdir backend
cd backend
# Inicializa o pacote NPM
npm init -y
Dentro deste diretório (backend):
criei o arquivo index.js
modifiquei o package.json
copiei um arquivo PDF chamado dummy.pdf
💡
O código da linha 8 à 16 trata de evitar problemas de CORS (Cross-Origin Resource Sharing) no navegador, já que estamos executando o frontend em localhost:4200 e o backend em localhost:3000. Se você não estiver familiarizado com CORS, recomendo a leitura deste link para entender melhor o conceito.
Para testar o código, utilizei um navegador para realizar uma requisição GET em localhost:3000 e, em seguida, utilizei o comando curl para testar o método POST no mesmo endereço.
Agora que está tudo certo no backend, vamos criar nosso projeto Angular.
Criando o projeto frontend com Angular
Para criar o projeto:
ng new --inline-template false --skip-git --skip-tests --minimal --strict frontend-pdf
💡
Se quiser saber mais sobre os parâmetros do comando acima, digite no seu terminal o seguinte: ng new help
Para testar o projeto:
cd frontend-pdf
npm run start
Utilizando o HttpClient para efetuar a requisição
Agora que o projeto está criado, vou importar o módulo do HttpClient e realizar um POST para ver o que acontece:
O resultado no navegador será:
Por padrão, o HttpClient espera que a resposta da requisição seja um JSON válido. Se a resposta não for um JSON válido, o HttpClient lançará o erro "Failure during parsing for http://localhost:3000". Como nossa API não retorna um JSON, mas sim bytes de um arquivo PDF, é necessário especificar o tipo de resposta esperada por meio do parâmetro responseType:
Executando a requisição novamente, temos o seguinte resultado:
Mostrando o PDF em um iframe
Agora que temos o PDF acessível, isto é, o conteúdo do PDF está armazenado na variável response, podemos criar uma URL exclusiva dentro do navegador utilizando o método window.URL.createObjectURL(). Podemos então vincular essa URL ao conteúdo do PDF e atribuí-la a um elemento <iframe>.
💡
O createObjectURL é um método JavaScript que cria um URL temporário para objetos de dados, como arquivos Blob ou File. Esse URL temporário permite que você acesse os dados desses objetos como se fossem recursos de URL, o que pode ser útil para exibir imagens, vídeos e outros tipos de mídia diretamente no navegador, sem a necessidade de upload para um servidor. É comumente utilizado em aplicações da web para pré-visualização de arquivos ou para trabalhar com conteúdo dinâmico.
Veja a implementação a seguir:
Resultado obtido em dois navegadores, Firefox Developer Edition e Safari:
Efetuando o download do PDF
Uma outra alternativa para disponibilizar o PDF ao usuário é o download. Então ao invés de exibir o conteúdo do PDF em um elemento específico, criei dinamicamente um elemento (<a>) e atribuí o valor da URL temporária ao atributo "href". Para garantir o download automático quando o elemento <a> for clicado, adicionei o atributo "download", fornecendo como valor o nome sugerido para o arquivo. Finalmente, realizei um clique programático para iniciar o download.
Confira a implementação abaixo:
Considerações
Este texto demonstra como podemos adquirir e compartilhar um arquivo PDF através de uma API. No entanto, esses princípios são facilmente aplicáveis a outros formatos de arquivo, como XLS, CSV, JPG, e muitos outros.
Além de explorarmos o uso do responseType do HttpClient, também empregamos o window.URL.createObjectURL para a criação de URLs de objetos.
É relevante mencionar que, no contexto de arquivos PDF, existem outras opções para exibição direta no frontend, como a biblioteca PDF.js, que pode ser uma alternativa viável.