Como Criar Componentes Globais e Dinâmicos com Strategy Pattern em React

Douglas Pujol - Oct 25 - - Dev Community

Tínhamos um desafio interessante no nosso time: como desenvolver componentes de forma global, onde os dados e comportamentos seriam diferentes para cada país? Em uma das discussões com meus colegas de trabalho, André Peixoto e Guilherme Cardoso, algumas soluções surgiram em nossa mente, e hoje vou contar sobre uma que se adequou perfeitamente ao nosso contexto.

No nosso caso, o primeiro componente que implementamos com essa abordagem foi uma tabela dinâmica. Esse componente deveria exibir colunas e dados que variavam conforme o país, com regras de exibição, interações e tratamentos de dados específicos.

O Desafio: Uma Tabela Dinâmica para Cada País

Imagine que você tem uma tabela que precisa renderizar diferentes tipos de dados para cada país, variando o número de colunas, o formato dos dados, e até as interações dos usuários com ela. Como resolver isso?

Você teria algumas opções, e vamos falar sobre algumas delas:

  1. Criar uma tabela para cada país. Essa abordagem poderia funcionar, mas traria alguns problemas:
    • Código duplicado: Cada tabela para cada país levaria à duplicação de lógica, aumentando o esforço de manutenção.
    • Dificuldade em sincronizar mudanças: Se um bug for encontrado, você precisaria corrigir em múltiplos lugares.
    • Escalabilidade: À medida que novos países fossem adicionados, essa solução se tornaria insustentável.
  2. Criar uma tabela única com condicionais. Outra abordagem seria desenvolver uma única tabela, com várias condicionais para renderizar cada tipo de célula ou header, dependendo do país. No entanto, isso também apresenta problemas:
    • Código complexo: À medida que a tabela cresce, o número de condicionais aumenta, tornando o código difícil de entender e manter.
    • Acoplamento: O código ficaria altamente acoplado, dificultando a adição de novas funcionalidades.
    • Manutenção difícil: A manutenção se torna um pesadelo quando o projeto cresce e precisa ser ajustado para novos cenários.

A Solução: Strategy Pattern

Uma abordagem mais limpa e escalável que adotamos foi o uso do Strategy Pattern. Mas o que é o Strategy Pattern?

O Strategy Pattern é um padrão de design comportamental que define uma família de algoritmos e os encapsula de maneira que possam ser intercambiáveis. Ou seja, você pode alternar entre diferentes estratégias sem modificar o cliente que as utiliza.

Benefícios do Strategy Pattern

Implementar um componente utilizando o Strategy Pattern traz algumas vantagens importantes:

  1. Separação de responsabilidades: Cada país tem sua própria estratégia para renderizar os dados, tornando o código mais modular e fácil de entender.
  2. Redução de código duplicado: Não precisamos replicar a lógica da tabela para cada país, já que as diferenças são abstraídas em estratégias específicas.
  3. Escalabilidade: Novos países ou mudanças nos existentes podem ser adicionados facilmente, sem impactar o código base.
  4. Facilidade de manutenção: Se houver um bug ou nova regra para um país, basta ajustar sua estratégia específica, sem impactar o restante da aplicação.

Pontos Negativos do Strategy Pattern

Embora o Strategy Pattern seja extremamente útil, ele não está isento de desvantagens. Uma consideração importante é a necessidade de configurar múltiplas estratégias separadas, o que pode resultar em um certo grau de fragmentação. Em sistemas maiores, pode ser necessário assegurar que as estratégias sejam mantidas de maneira organizada e coesa.

Implementação: Nossa Tabela Dinâmica

Agora, vamos ver como a tabela foi implementada na prática usando o Strategy Pattern.

Primeiro, criamos um componente Table que recebe os dados e a estratégia específica de colunas para renderização:

import React from 'react';

export interface Column<T> {
  key: string;
  label: string;
  render: (row: T) => JSX.Element;
}

export interface TableProps<T = unknown>  {
  data?: T[];
  strategy?: {
    columns: Column<T>[];
  } | unknown ;
}

const Table =<T,>({ data, strategy }: TableProps<T>) => {

  const defaultStrategy = { columns: [] as Column<T>[] };
  const currentStrategy = strategy ? (strategy as { columns: Column<T>[] }) : defaultStrategy;

  return (
    <table>
      <thead>
        <tr>
          {currentStrategy?.columns?.map((col) => (
            <th key={col.key} className="px-4 py-2 border border-dashed border-border">
              {col.label}
            </th>
          ))}
        </tr>
      </thead>
      <tbody>
        {currentStrategy && data && data.length > 0  ? data?.map((row, index) => (
          <tr key={index}
            className="hover:bg-hoverTable"
          >
            {currentStrategy?.columns?.map((col) => col.render(row))}
          </tr>
        )) : (
          <tr>
            <td colSpan={currentStrategy?.columns?.length || 1} className="px-4 py-2 text-center">
              No data available
            </td>
          </tr>
        )}
      </tbody>
    </table>
  );
};

export default  React.memo(Table);

Enter fullscreen mode Exit fullscreen mode

Com essa estrutura básica, conseguimos garantir que as colunas e o conteúdo sejam renderizados dinamicamente com base na estratégia específica do país. Para cada país, definimos uma estratégia diferente, como no exemplo abaixo:

export const tableStrategies = {
  BR: {
    columns: [
      {
        key: 'number',
        label: 'Número',
        render: (row: BrazilData): JSX.Element => (
          <td key={`${row.number}-br`} className="px-4 py-2 border border-dashed border-border">{row.number}</td>
        ),
      },
      {
        key: 'name',
        label: 'Nome',
        render: (row: BrazilData): JSX.Element => (
          <td key={`${row.number}-name`} className="px-4 py-2 border border-dashed border-border">{row.name}</td>
        ),
      },
      // ...outras colunas
    ] as BrazilColumn[],
  },
  ES: {
    columns: [
      {
        key: 'shirtNumber',
        label: 'Número de Camiseta',
        render: (row: SpainData): JSX.Element => (
          <td key={`${row.shirtNumber}-es`} className="px-4 py-2 border border-dashed border-border">{row.shirtNumber}</td>
        ),
      },
      {
        key: 'name',
        label: 'Nombre',
        render: (row: SpainData): JSX.Element => (
          <td key={`${row.name}-name`} className="px-4 py-2 border border-dashed border-border">{row.name}</td>
        ),
      },
      // ...outras colunas
    ] as SpainColumn[],
  },
};

Enter fullscreen mode Exit fullscreen mode

Além disso, o uso do método render no contexto do Strategy Pattern traz simplicidade. Cada coluna da tabela possui uma função render, que é responsável por determinar como a célula será exibida para cada tipo de dado específico. Isso oferece grande flexibilidade e clareza para personalizar a exibição sem complexidade desnecessária.

Agora, basta chamar o componente Table com a estratégia apropriada e os dados para cada país:

<Table
  strategy={tableStrategies.BR}
  data={brazilData}
/>
Enter fullscreen mode Exit fullscreen mode

Um Exemplo Simples com Seleção de Países

Para fins didáticos, no projeto criei uma simulação onde o usuário pode selecionar diferentes países através de um select. Dependendo da escolha do país, os dados e a estratégia de renderização da tabela são atualizados dinamicamente:

import React from 'react';
import {  CountryCode } from './types';
import  Table  from './components/Table';
import { americanData, brazilData, southKoreaData, spainData } from './mocks';
import { tableStrategies } from './strategies';

export default function Home() {

  const [strategy, setStrategy] = React.useState<unknown>({});
  const [selectedCountry, setSelectedCountry] = React.useState<CountryCode>('BR');
  const [data, setData] = React.useState<unknown[]>([null])

  const updateDataAndStrategy =React.useCallback((country: CountryCode) => {
    switch (country) {
      case 'BR':
        setData(brazilData);
        setStrategy(tableStrategies[country]);
        break;
      case 'US':
        setData(americanData);
        setStrategy(tableStrategies[country]);
        break;
      case 'ES':
        setData(spainData);
        setStrategy(tableStrategies[country]);
        break;
      case 'KR':
        setData(southKoreaData);
        setStrategy(tableStrategies[country]);
        break;
      default:
        setData([]);
        setStrategy({});
        break;
    }
  }, []);

  const handleChanged = (event: React.ChangeEvent<HTMLSelectElement>) => {
    const selected = event.target.value as CountryCode;
    setSelectedCountry(selected);
    updateDataAndStrategy(selected);
  };

  return (
    <>
      <h1>## Table</h1>
      <select onChange={handleChanged} value={selectedCountry} className="mb-2">
        <option value="US">United States</option>
        <option value="AR">Argentina</option>
        <option value="BR">Brazil</option>
        <option value="KR">South Korea</option>
        <option value="ES">Spain</option>
      </select>

      <Table
        strategy={strategy}
        data={data}
       />
    </>
  );
}

Enter fullscreen mode Exit fullscreen mode

Nesse exemplo, o componente select permite ao usuário selecionar o país desejado, e a tabela atualiza automaticamente para exibir os dados e a estratégia de renderização específicos daquele país. Confira o Link do projeto

foto da tabela completada do meu projeto no github

Alternando Entre Países

Com o seletor de país, trocamos facilmente entre diferentes estratégias, simulando a renderização personalizada da tabela de acordo com as configurações específicas de cada país. Essa abordagem permite visualizar, em tempo real, como o conteúdo se adapta a diferentes regiões e contextos.

foto da minha tabela no github, estou trocando os países para mostrar o strategy pattern

Conclusão

Com essa abordagem utilizando Strategy Pattern, garantimos um código mais limpo, modular e escalável. Além disso, se no futuro precisarmos adicionar novos países ou ajustar as regras de um país existente, podemos fazer isso facilmente sem impactar o restante do sistema.

Eu desenvolvi um projeto no GitHub que contém todo esse código e um exemplo prático de como aplicar essa técnica.

Se quiser ver o código completo, confira o repositório no GitHub.

Valeu galera :)

. . . . . .
Terabox Video Player