Pular para o conteúdo principal

React - Elevando o state (state lift)

Como visto anteriormente, o fluxo de informações no React é unidirecional, partindo sempre do componente pai para os filhos, logo um componente pai desconhece o que acontece com o estado presente nos seus componentes filhos.

Então como é que nós fazemos quando precisamos que com que um componente filho produza ou modifique um valor para que esse possa ser utilizado pelo componente pai? Bom a resposta para essa pergunta esta no titulo da postagem, resolvemos essa questão com a elevação do "state".

A elevação do "state" em tese é algo relativamente simples, basta mudar o estado do filho para o pai, logo o filho não mais trabalhará com seu estado, ele receberá o estado do pai por meio das "props" e quando for necessário alterar o estado o componente filho deve chamar um método do pai, que também estará presente em "props".

Exemplo

No exemplo abaixo criaremos um componente "filho" que oferece uma lista de opções de acordo com um tipo, ao escolher um item esse valor deve ser retornado para o componente pai:

import React, { Component } from 'react'

class Lista extends Component {
       
    handleClick(codigo) {
        this.props.onOptionChange(codigo);
    }

    render() {
        return(
            <ul>
                {this.props.lista === undefined 
<li /> 
this.props.lista.map((item=>
                    <li 
                        key={item.codigo} 
                        onClick={this.handleClick.bind(thisitem.codigo)}
                        value={item.nome}
                        >                        
                        {item.nome}
                    </li>
                )}
            </ul>        
            );            
    }
}

export default Lista

No componente "Lista" recebemos um "array" que é mostrado para seleção, sempre que um item é clicado o mesmo é incluído em uma outra lista no pai.

import React from 'react';
import logo from './logo.svg';
import './App.css';
import Lista from './Lista';

class App extends React.Component {

  listaValores = [
    {codigo: 1007nome:"Memoria ram"tipo:"PC"},
    {codigo: 1009nome:"Teclado sem fio"tipo:"PC"},
    {codigo: 1012nome:"Monitor"tipo:"PC"},
    {codigo: 1109nome:"Saia"tipo:"VE"},
    {codigo: 1112nome:"Vestido"tipo:"VE"}
  ];

  constructor(props) {
    super(props);
    this.state = {
      list: []
    };

    this.onOptionChange = this.onOptionChange.bind(this);
  }

  onOptionChange(item) {
    this.setState(state => {
      const list = state.list.concat(item);

      return {
        list
      };
    });
  }

  render() {
    return (
      <div className="App">
        <header className="App-header">
          <img src={logo} className="App-logo" alt="logo" />
          
          <div>
            Escolhidos:
            {this.state.list.toString()}
          </div>

          <div>
            Informática:
            <Lista lista={this.listaValores.filter(
function(item){ return item.tipo === 'PC'})} 
onOptionChange={this.onOptionChange} >
</Lista>
          </div>

          <div>
            Vestuário:
            <Lista lista={this.listaValores.filter(
function(item){ return item.tipo === 'VE'})} 
onOptionChange={this.onOptionChange} >
</Lista>
          </div>

        </header>
        
      </div>
    );
  }
}

export default App;

No componente "App" usamos o componente "Lista" duas vezes, uma com informações sobre informática e outra com informações sobre vestuário, sempre que o componente é usado a coleção de selecionados do pai é atualizada.

O código de exemplo pode ser encontrado aqui.

Comentários

Mais visitadas

Listar arquivos existentes em diretório (Delphi)

Mostraremos uma maneira simples e prática para listar o conteúdo de um diretório com a opção de incluir nessa listagem os arquivos de seus subdiretórios. No exemplo abaixo temos um Edit para receber o diretório a ser pesquisado um CheckBox para indicar se os subdiretórios entrarão na pesquisa um botão para efetuar a pesquisa e um Memo para listar os arquivos encontrados, no final um Edit que receberá o cálculo final (em bytes) da soma do tamanho dos arquivos. procedure TForm1.Button1Click(Sender: TObject); begin   tamanhoTotal := 0;   memLista.Lines.Clear;   ListarArquivos(edtDiretorio.Text, chkSub.Checked);   Edit1.Text := IntToStr( tamanhoTotal ); end; procedure TForm1.ListarArquivos(Diretorio: string; Sub:Boolean); var   F: TSearchRec;   Ret: Integer;   TempNome: string; begin   Ret := FindFirst(Diretorio+'\*.*', faAnyFile, F);   try     while Ret = 0 do ...

Alterar cores do PageControl (Delphi)

O padrão Windows todo cinza não é muito atraente, por isso quando nos utilizarmos do PageControl podemos alterar suas cores e fontes da seguinte maneira: Em primeiro lugar devemos alterar a propriedade OwnerDraw para TRUE ; Depois implementar seu método DrawTab da seguinte maneira: //pinta a fonte Control.Canvas.Font.Color:=clBlack; // // pinta a paleta / aba Control.Canvas.brush.Color:=clSkyBlue; PageControl1.Canvas.Rectangle(Rect); Control.Canvas.TextOut(Rect.left+5,Rect.top+3,PageControl1.Pages[tabindex].Caption); // pinta a parte interna (tabsheet) PageControl1.Pages[TabIndex].brush.Color := Control.Canvas.brush.Color; PageControl1.Pages[TabIndex].Repaint; Caso a intenção seja manter cada aba com seu próprio estilo basta adicionar um CASE filtrando o índice das abas: case TabIndex of   0: Control.Canvas.Font.Color:=clBlack;   1: Control.Canvas.Font.Color:=clWindow; ...

Como Verificar se um Objeto Existe (Delphi)

Em alguns momentos surge a necessidade de verificar se um determinado objeto existe, ou seja se já foi criado, principalmente quando se trabalha com criação dinâmica em tempo de execução, então vamos ao exemplo: - Vamos criar uma variável, um vetor do tipo caixa de texto: var Minha_caixa : array of TEdit; - Em seguida definir o tamanho desse vetor, no caso será dez: setLength(Minha_caixa, 10) - Agora iremos criar nossa caixa de texto: // lembrando que o vetor inicia em zero // logo o índice final é o tamanho total - 1 for vl_i := 0 to Length(Minha_caixa) -1 do begin Minha_caixa[vl_i] := TEdit.Create(self); with Minha_caixa[vl_i] do begin Parent := Self; Name := 'Caixa_N'+IntToStr(vl_i); Text := 'Esta é a '+IntToStr(vl_i)+' º caixa !'; ReadOnly := true; Height := 21; Width := ...