使用react-native和recoil创建TodoList,发生问题

332nm8kg  于 2023-03-31  发布在  React
关注(0)|答案(1)|浏览(147)

这是我的代码但是,在输入一个值并将该值传递给按钮后,发生了此错误。
Error: Too many re-renders. React limits the number of renders to prevent an infinite loop.
但是,我删除了不断变化的状态,它工作了。
我不知道我错过了什么。帮帮我

/**
 * Sample React Native App
 * https://github.com/facebook/react-native
 *
 * @format
 * @flow strict-local
 */

import { Alert } from 'react-native';
import {
  SafeAreaView,
  ScrollView,
  StatusBar,
  StyleSheet,
  TextInput,
  Button,
  useColorScheme,
  View,
} from 'react-native';

import {
  Colors,
} from 'react-native/Libraries/NewAppScreen';
import { RecoilRoot, useRecoilValue, useRecoilState, useSetRecoilState } from 'recoil';
import { selectStatus, selectToDo, toDos } from './store/toDo';
import { useState } from 'react';

const Apps = () => {

  const toDoAtom = useRecoilValue(toDos);
  const setToDoAtom = useSetRecoilState(toDos)
  const [nowStatus, setStatus]= useRecoilState(selectStatus)
  const [toDo, onChangeToDo] = useState('')
  const [selectToDos, setNewToDos] = useRecoilState(selectToDo);
  const [id, setId] = useState(1)

  const isDarkMode = useColorScheme() === 'dark';

  const backgroundStyle = {
    backgroundColor: isDarkMode ? Colors.darker : Colors.lighter,
  };

  const onSetData = () => {
    const newToDoList = [
      ...toDoAtom,
      {
        id:id,
        contents:toDo,
        status: "DOING"
      }
    ]
    setToDoAtom(newToDoList)
    setId((id)=>id+1)
    onChangeToDo("")
    Alert.alert('submit')
  }

  const onChangeStatue = (data) => {

    const targetIndex = data.id
    var lists = [
      ...toDoAtom.slice(0, targetIndex),
      {id:data.id, contents:data.contents, status:"DONE"},
      ...toDoAtom.slice(targetIndex + 1)
    ]

    setToDoAtom(lists)

  }
  const onRest = () => {
    console.log(toDoAtom)
    setNewToDos([])
    setId(1)
  }

  const onStatus = () => {
    if(nowStatus == "DOING"){
      setStatus("DONE")
    }else if(nowStatus == "DONE"){
      setStatus("DOING")
    }
  }

  return (
    <RecoilRoot>
      <SafeAreaView style={backgroundStyle}>
        <StatusBar barStyle={isDarkMode ? 'light-content' : 'dark-content'} />
        <ScrollView
          contentInsetAdjustmentBehavior="automatic"
          style={backgroundStyle}>
          <View
            style={{
              backgroundColor: isDarkMode ? Colors.black : Colors.white,
            }}>
              <View>
              <TextInput
                style={styles.input}
                onChangeText={onChangeToDo}
                value={toDo}
              />
              <Button title='submit' color="blue" onPress={onSetData} ></Button>
              <Button
                title = "Reset"
                color = "red"
                onPress={onRest}
              />
              </View>
              <Button title={nowStatus} onPress={onStatus}/>
                {selectToDos?.map((hi)=>(
                  <View key={hi.id}>
                    <Button title={hi.contents} onPress={onChangeStatue(hi)} ></Button>
                  </View>
                ))} 
          </View>
        </ScrollView>
      </SafeAreaView>
    </RecoilRoot>
  );
};

export const App = () => {
  return(
    <RecoilRoot>
      <Apps/>
    </RecoilRoot>
  )
}

const styles = StyleSheet.create({
  input: {
    height: 40,
    margin: 12,
    borderWidth: 1,
    padding: 10,
  },
});

export default App;
import { selector, atom } from 'recoil'

export const status = atom({
    key:'status',
    default:'DOING'
})

export const toDos = atom({
    key:'toDos',
    default:[]
})

export const selectToDo = selector({
    key:'selectToDo',
    get: ({ get }) => {
        const originalToDos = get(toDos);
        const nowStatus = get(status)
        return originalToDos.filter((toDo) => toDo.status === nowStatus);
    },
})

export const selectStatus = selector({
    key:'selectStatus',
    get : ({get}) => {
        return get(status)
    },
    set: ({set}, selectStatus) => {
        set(status, selectStatus)
    }
})

在使用过滤器选择状态并显示值之后,我尝试将Doing状态更改为Done。

5gfr0r5j

5gfr0r5j1#

你有一个功能组件,它在每个状态或属性改变时呈现。每次呈现时,按钮的onPress方法在呈现时调用状态改变函数。

<View key={hi.id}>
 <Button title={hi.contents} onPress={onChangeStatue(hi)} ></Button>
</View>

onPress应该接收一个***可调用的***函数,而不是被调用的函数:

<View key={hi.id}>
 <Button title={hi.contents} onPress={() => onChangeStatue(hi)} ></Button>
</View>

相关问题