上下文状态被意外更新 - React Typescript

时间:2021-04-27 11:33:58

标签: javascript reactjs api state context-api

第一次发布海报,如果需要更多信息,请告诉我。

试图弄清楚为什么即使我的 setSate 方法被注释掉,我使用上下文 API 的全局状态也会被更新。我以为我可能是不小心直接改变了状态,但我不相信我是

actionOnClick() 中的“specialModes”是有问题的状态

const SpecialFunctions: FC = (props: Props) => {
const { currentModeContext, specialModesContext: specialActionsContext, performCalc, inputValueContext } = useContext(AppContext)
const { specialModes, setSpecialModes } = specialActionsContext
const { currentMode, setCurrentMode } = currentModeContext
const { inputValue, setInputValue } = inputValueContext

const categoryOnClick = (index: number) => {
    setCurrentMode(specialModes[index])
    console.log(specialModes[index].title);
}

const actionOnClick = (action: IAction) => {
    let newAction = action
    newAction.value = performCalc()
    let newSpecialModes = specialModes.map((mode) => {
        if (mode === currentMode) {
            let newMode = mode
            newMode.actions = mode.actions.map((element) => {
                if (element === action) {
                    return newAction
                }
                else return element
            })
            return newMode
        }
        else return mode

    })
    //setSpecialModes(newSpecialModes)

}

let headings = specialModes.map((categorgy, index) => {
    return <Heading isActive={categorgy === currentMode ? true : false} onClick={() => categoryOnClick(index)} key={index}>{categorgy.title}</Heading>
})

let actions = currentMode.actions.map((action, index) => {
    return (
        <Action key={index} onClick={() => actionOnClick(action)}>
            <ActionTitle>{action.title}</ActionTitle>
            <ActionValue>{action.value}</ActionValue>
        </Action>
    )
})
return (
    <Wrapper>
        <Category>
            {headings}
        </Category>
        <ActionsWrapper toggleRadiusCorner={currentMode === specialModes[0] ? false : true}>
            {actions}
        </ActionsWrapper>

    </Wrapper>
)

}

Context.tsx

interface ContextType {
specialModesContext: {
    specialModes: Array<ISpecialModes>,
    setSpecialModes: React.Dispatch<React.SetStateAction<ISpecialModes[]>>
},
currentModeContext: {
    currentMode: ISpecialModes,
    setCurrentMode: React.Dispatch<React.SetStateAction<ISpecialModes>>
},
inputValueContext: {
    inputValue: string,
    setInputValue: React.Dispatch<React.SetStateAction<string>>
},
inputSuperscriptValueContext: {
    inputSuperscriptValue: string,
    setInputSuperscriptValue: React.Dispatch<React.SetStateAction<string>>
},
performCalc: () => string
}

export const AppContext = createContext({} as ContextType);

export const ContextProvider: FC = ({ children }) => {
const [SpecialModes, setSpecialModes] = useState([
    {
        title: 'Rafter',
        actions: [
            {
                title: 'Span',
                active: false
            },
            {
                title: 'Ridge Thickness',
                active: false
            },
            {
                title: 'Pitch',
                active: false
            }
        ],
    },
    {
        title: 'General',
        actions: [
            {
                title: 'General1',
                active: false
            },
            {
                title: 'General2',
                active: false
            },
            {
                title: 'General3',
                active: false
            }
        ],
    },
    {
        title: 'Stairs',
        actions: [
            {
                title: 'Stairs1',
                active: false
            },
            {
                title: 'Stairs2',
                active: false
            },
            {
                title: 'Stairs3',
                active: false
            }
        ],
    }
] as Array<ISpecialModes>)

const [currentMode, setCurrentMode] = useState(SpecialModes[0])

const [inputValue, setInputValue] = useState('0')
const [inputSuperscriptValue, setInputSuperscriptValue] = useState('')

const replaceCharsWithOperators = (string: string): string => {
    let newString = string.replaceAll(/\s/g, '') // delete white space 
    newString = newString.replace('×', '*')
    newString = newString.replace('÷', '/')
    console.log(string)
    console.log(newString)
    return newString
}

const performCalc = (): string => {
    let originalEquation = `${inputSuperscriptValue} ${inputValue} =`
    let equation = inputSuperscriptValue + inputValue
    let result = ''
    equation = replaceCharsWithOperators(equation)
    result = eval(equation).toString()
    setInputSuperscriptValue(`${originalEquation} ${result}`)
    setInputValue(result)
    console.log(result)
    return result
}

return (
    <AppContext.Provider value={
        {
            specialModesContext: {
                specialModes: SpecialModes,
                setSpecialModes: setSpecialModes
            },
            currentModeContext: {
                currentMode,
                setCurrentMode
            },
            inputValueContext: {
                inputValue,
                setInputValue
            },
            inputSuperscriptValueContext: {
                inputSuperscriptValue,
                setInputSuperscriptValue
            },
            performCalc
        }}>
        {children}
    </AppContext.Provider>
)
}

1 个答案:

答案 0 :(得分:0)

在您的 mode.actions.map() 函数中,您间接更改了原始 actions 数组的 specialModes 字段。

要解决此问题,您需要使用 specialModes ES6 扩展运算符创建 ... 数组的浅拷贝。

const clonedSpecialModes = [...specialModes];
let newSpecialModes = clonedSpecialModes.map((mode) => { 
    // rest of your logic here
})