Como exportar dados de uma API para Excel Usando React

react to excel

Vamos começar iniciando um projeto em React, usando nosso bom e velho CRA

npx create-react-app react-excel

Para poder nos ajudar e facilitar bastante o nosso trabalho, vamos usar a lib sheetJS CE deixarei o link da documentação deles aqui, se quiserem dar uma olhada, é uma documentação ótima.

Para instalar no nosso projeto é bem simples:

// npm
npm i --save https://cdn.sheetjs.com/xlsx-0.20.1/xlsx-0.20.1.tgz

// pnpm
pnpm install --save https://cdn.sheetjs.com/xlsx-0.20.1/xlsx-0.20.1.tgz

// yarn
yarn add https://cdn.sheetjs.com/xlsx-0.20.1/xlsx-0.20.1.tgz

Trabalharemos dentro do nosso App.js mesmo, pois a implementação é bem simples, mas pensando em React sempre precisamos ter em mente que esse deve ser um componente que pode ser reutilizável em outros locais da sua aplicação.

Vamos modificar o nosso App.js para deixar dessa forma:

import './App.css';

function App() {
  const handleDownload = () => {
    console.log('Downloading')
  }


  return (
    <div className="App">
      <button onClick={handleDownload}>Download</button>
    </div>
  );
}

export default App;

Vamos usar uma API de dados fake, você pode usar qualquer uma que você conhece, mas se não conhecer nenhuma, estou usando essa aqui, JSONPLACEHOLDER. Daí o código fica dessa forma.

import { useEffect, useState } from "react";
import "./App.css";

function App() {
  const [posts, setPosts] = useState([]);

  useEffect(() => {
    fetch('https://jsonplaceholder.typicode.com/posts')
      .then(response => response.json())
      .then(json => setPosts(json))
  }, []);

  const handleDownload = () => {
    console.log(posts);
  };
  return (
    <div className="wrapper">
      <button onClick={handleDownload}>DOWNLOAD EXCEL</button>
    </div>
  );
}

export default App;

Agora vamos implementar dentro da função handleDownload nossa lógica para adicionarmos os dados dentro de um arquivo de excel e fazer o download desse arquivo.

import { useEffect, useState } from "react";
import "./App.css";
import * as XLSX from "xlsx";

function App() {
  const [posts, setPosts] = useState([]);

  useEffect(() => {
    fetch('https://jsonplaceholder.typicode.com/posts')
      .then(response => response.json())
      .then(json => setPosts(json))
  }, []);

  const handleDownload = () => {
    // Aqui criamos as rows da tabela
    const rows = posts.map((post) => ({
      id: post.userId,
      title: post.title,
      body: post.body,
    }));

    // cria workbook e worksheet
    const workbook = XLSX.utils.book_new();
    const worksheet = XLSX.utils.json_to_sheet(rows);

    XLSX.utils.book_append_sheet(workbook, worksheet, "Posts");

    XLSX.writeFile(workbook, "posts.xlsx", { compression: true });
  };
  return (
    <div className="wrapper">
      <button onClick={handleDownload}>DOWNLOAD EXCEL</button>
    </div>
  );
}

export default App;

Destaquei que na const rows criamos as linhas da tabela, e id, title e body, serão os headers da nossa tabela, e as consts workbook e worksheet criamos uma instância para o nosso arquivo de excel e depois convertemos um estrutura json para uma tabela. Por último críamos file usando XLXS.writeFile passamos o workbook, passamos o nome do arquivo, que no nosso caso vai ser posts.xlsx se for excel do Windows seria posts.xls e podemos deixar o arquivo comprimido ou não.

Se tudo deu certo você irá conseguir baixar um arquivo dessa forma: planilha de excel

É isso, bem simples não ? Espero que tenham gostado, qualquer sugestão ou feedback será muito bem vindo. Obrigado.

A minha dúvida é: porque colocar um processamento pesado desse no browser e não no servidor?

Cara no meu projeto atual, tive que implementar isso no front, que não é nada legal, ou seja, podemos enfrentar isso em vários tipos de projetos.

Interessante, mas arriscado colocar esse processamento em um ponto de chegada dos dados(cliente) somente.

Sim. Tudo tem o seu lado bom e ruim, eu mesmo prefiro que o processamento de dados não seja no lado do cliente.